0

我正在尝试实现一个解析基本算术表达式的简单 DSL。这需要在浏览器中完成,所以我使用 PEGjs 来生成解析器。

表达式中的术语可以是数字(整数或实数)、变量(变量是传递给解析器的上下文对象的属性)、条件或通过点符号访问的属性。

我希望条件看起来像这样condition?value,如果condition为真,则该术语等同于value。两侧的变量?也可以是点符号访问的对象的属性,例如 this object.property1?object.property2

因此,如果向解析器传递了一个像这样的对象:

context = {
  depth: 100,
  material: {
    thickness: 20
    include: true
  }
  edge: {
    face: 4.5
  }
}

表达方式:

500 + depth + material.include?edge.face + material.thickness应等于624.5

我一直在使用 PEGjs 在线编辑器。我尝试了很多不同的方法,但我似乎无法确定条件。其他一切都有效。以下是相关规则:

Variable "variable"
  = variable:identifier accessor:("." identifier)* {
      var result = context[variable], i

      for (i = 0; i < accessor.length; i++) {
        result = result[accessor[i][1]]
      }

      return result
    }

identifier
  = identifier:$([0-9a-zA-Z_\$]+)

Conditional
  = condition:Variable "?" value:Variable {
    return condition ? value : 0
  }

我在 PEGjs github repo 中查看了 javascript 的示例语法,条件规则看起来很像我在这里得到的,但我仍然无法让它工作。

像我在 PEGjs 规则中描述的那样,实现条件语句的正确方法是什么?

4

1 回答 1

1

我知道这有点晚了,但问题是 yourvariable是一个评估为"material.include".

看看这段代码:

var result = context[variable], i

您正在尝试从上下文对象中访问名为“material.include”的属性,如下所示:

{
    "material.include": true
}

而不是尝试访问“material”属性引用的对象,然后访问结果对象的“include”属性,如下所示:

{
    "material": {
        "include": true
    }
}

解决方案是按"."字符拆分变量字符串,然后递归查找您的属性:

Variable "variable"
  = variable:identifier accessor:("." identifier)* {
      var path = variable.split(".");
      var result = path.reduce( (nextObject, propName) => nextObject[propName], context );

      for (var i = 0; i < accessor.length; i++) {
        result = result[accessor[i][1]]
      }

      return result
    }

请注意,此解决方案并不完整,因为如果您尝试访问未在上下文中定义的material.include位置,则会导致错误。material您可能想要添加额外的错误处理,但它确实适用于给定的示例。

于 2020-02-21T15:57:29.650 回答