从JSON对象解析+评估表达式而不使用Eval()?

时间:2013-07-04 03:23:34

标签: javascript node.js parsing

我正在构建一个节点应用程序,用户可以(理想情况下)使用一系列JSON对象为地理数据定义样式:

{
    "style":
        {
            "test": "year",
            "condition": "<= 1954 AND >= 1936",
            "color": "red"
        }
}

在上面的例子中,我喜欢将该样式评估为

if (year <= 1954 && year >= 1936){
    object.color = red;
}

有没有一种简单的方法来解析+评估这样的表达式/从这样的对象构建它们?我特别感兴趣的是让人们将使用&lt; =,&gt; =,||和&amp;&amp ;;等。

如果可能的话,我想避免使用eval()。

2 个答案:

答案 0 :(得分:5)

如果您不想使用eval,则必须编写自己的小解析器并创建如下定义语言:

"condition": ["and", ["<=", 1954], [">=", 1936]],

这是您可以考虑的部分实施:

function do_and(args, value)
{
  for (var i = 0; i < args.length; ++i) {
    if (!evaluate(args[i], value)) {
      return false;
    }
  }
  return true;
}

function evaluate(condition, value)
{
  switch (condition[0]) {
    case "and":
      return do_and(condition.slice(1), value);

    case "<=":
      return value <= condition[1];

    case ">=":
      return value >= condition[1];
  }
}

这就是你如何使用它:

var style = {
    "test": "year",
    "condition": ["and", ["<=", 1954], [">=", 1936]],
    "color": "red"
}, context = {
  "year": 1940
};    

if (evaluate(style.condition, context[style.test])) {
  console.log(style.color); // "red"
}

Demo

答案 1 :(得分:1)

这样的东西
var obj = JSON.parse(str);
switch (obj.style.operator){
    case '=':
    if (window[obj.style.condition] === obj.style){//assuming that the conditions are global
        object.color = obj.style;
    }
    break;
    ...
}