Инфиксный формат для макроса Nemerle - PullRequest
3 голосов
/ 22 октября 2008

Скажем, мне нужен какой-то особенный оператор умножения. Это может быть реализовано в следующем макросе:

macro @<<!(op1, op2)
{
    <[ ( $op1 * $op2 ) ]>
}

И я могу использовать его как

def val = 2 <<! 3

И его работа.

Но что мне действительно нужно, так это некоторый «английский» оператор для DSL, который я сейчас разрабатываю:

macro @multiply(op1, op2)
{
    <[ ( $op1 * $op2 ) ]>
}

и если я попытаюсь использовать его как

def val = 2 multiply 3

компилятор завершается неудачно с ожидаемым; Ошибка

В чем проблема? Как я могу реализовать этот макрос формата infix?

Ответы [ 2 ]

7 голосов
/ 22 октября 2008

Прямо из исходного кода компилятора:

namespace Nemerle.English
{
  [assembly: Nemerle.Internal.OperatorAttribute ("Nemerle.English", "and", false, 160, 161)]
  [assembly: Nemerle.Internal.OperatorAttribute ("Nemerle.English", "or", false, 150, 151)]
  [assembly: Nemerle.Internal.OperatorAttribute ("Nemerle.English", "not", true, 181, 180)]  

  macro @and (e1, e2) {
    <[ $e1 && $e2 ]>
  }

  macro @or (e1, e2) {
    <[ $e1 || $e2 ]>
  }

  macro @not (e) {
    <[ ! $e ]>
  }

Вам нужно разбросать атрибуты OperatorAttributes, и он будет работать. Кстати, OperatorAttribute определяется следующим образом:

public class OperatorAttribute : NemerleAttribute
{
  public mutable env : string;
  public mutable name : string;
  public mutable IsUnary : bool;
  public mutable left : int;
  public mutable right : int;
}
1 голос
/ 22 октября 2008

Как обычно, я нашел ответ раньше, чем общение ответило :) Таким образом, решение состоит в том, чтобы просто использовать специальный атрибут уровня сборки, который определяет макрос как бинарный оператор:

namespace TestMacroLib
{
  [assembly: Nemerle.Internal.OperatorAttribute ("TestMacroLib", "multiply", false, 160, 161)]
  public macro multiply(op1, op2)
  {
    <[ ( $op1 * $op2 ) ]>
  }
}
...