使用NCalc在表达式中获取参数

时间:2014-04-11 13:09:58

标签: c# string expression ncalc

我有一个表达式,我想解析它以获取所有使用参数的列表。

  

例如:" X + 5 /(Y - 1)"应该给我以下结果:X,   ÿ

我已经在我的项目中使用了NCalc;那么可以使用NCalc来获取表达式中使用的参数吗?

根据这一个讨论条目(https://ncalc.codeplex.com/discussions/361959),它是,但我不太明白答案。

3 个答案:

答案 0 :(得分:12)

从讨论/答案:http://ncalc.codeplex.com/discussions/360990

我已经测试并运行的实现(针对您提供的示例表达式)是实现LogicalExpressionVisitor并让它记录下找到的参数:

class ParameterExtractionVisitor : LogicalExpressionVisitor
{
    public HashSet<string> Parameters = new HashSet<string>();

    public override void Visit(NCalc.Domain.Identifier function)
    {
        //Parameter - add to list
        Parameters.Add(function.Name);
    }

    public override void Visit(NCalc.Domain.UnaryExpression expression)
    {
        expression.Accept(this);
    }

    public override void Visit(NCalc.Domain.BinaryExpression expression)
    {
        //Visit left and right
        expression.LeftExpression.Accept(this);
        expression.RightExpression.Accept(this);
    }

    public override void Visit(NCalc.Domain.TernaryExpression expression)
    {
        //Visit left, right and middle
        expression.LeftExpression.Accept(this);
        expression.RightExpression.Accept(this);
        expression.MiddleExpression.Accept(this);
    }

    public override void Visit(Function function)
    {
        foreach (var expression in function.Expressions)
        {
            expression.Accept(this);
        }
    }

    public override void Visit(LogicalExpression expression)
    {
        expression.Accept(this);
    }

    public override void Visit(ValueExpression expression)
    {

    }
}

然后你会用它作为:

var expression = NCalc.Expression.Compile("2 * [x] ^ 2 + 5 * [y]", false);

ParameterExtractionVisitor visitor = new ParameterExtractionVisitor();
expression.Accept(visitor);

var extractedParameters = visitor.Parameters;

foreach (var param in extractedParameters)
    Console.WriteLine(param);

这输出&#34; x&#34;和&#34; y&#34;对我来说。

请注意HashSetParameterExtractionVisitor的使用情况。这是因为如果您的表达式多次包含相同的变量(例如:"[x] + [x]"),它将被添加两次。如果您想在每次使用相同变量时存储条目,请将HashSet替换为List


总而言之,我对NCalc的经验很少,所以我对LogicalExpressionVisitor的重写方法的实现是猜测。当我使用void Visit(ValueExpression expression)覆盖expression.Accept(this)方法时,会产生StackOverflowException。所以我只是将实现空白似乎就可以了。所以我建议你在这里用非常大的盐来回答我的答案。您的里程可能会有所不同,我不能说这是否适用于所有类型的表达。

答案 1 :(得分:2)

这对我有用。您的里程可能会有所不同。

   public List<string> GetParameters(string expression) {
       List<string> parameters = new List<string>();
       Random random = new Random();
       NCalc.Expression e = new NCalc.Expression(expression);

       e.EvaluateFunction += delegate(string name, NCalc.FunctionArgs args) {
           args.EvaluateParameters();
           args.Result = random.Next(0, 100);
       };
       e.EvaluateParameter += delegate(string name, NCalc.ParameterArgs args) {
           parameters.Add(name);
           args.Result = random.Next(0, 100);
       };
       try {
           e.Evaluate();
           }
       catch {
            }
       return parameters;
    }

参考:https://ncalc.codeplex.com/discussions/79258#editor

答案 2 :(得分:0)

这是我使用的另一种方法:

我构建了一个NCalc扩展方法,允许动态处理参数和函数。

internal static class NCalcExtensions
{
    public static object Evaluate(this Expression exp, EvaluateParameterHandler evaluateParameters = null, EvaluateFunctionHandler evaluateFunctions = null)
    {
        try
        {
            if (evaluateParameters != null)
                exp.EvaluateParameter += evaluateParameters;

            if (evaluateFunctions != null)
                exp.EvaluateFunction += evaluateFunctions;

            return exp.Evaluate();
        }
        finally
        {
            exp.EvaluateParameter -= evaluateParameters;
            exp.EvaluateFunction -= evaluateFunctions;
        }
    }
}

除此之外,我可以使用它来运行虚拟评估来获取参数和函数名称。

var paramNames = new List<string>();
var functionNames = new List<string>();

expression.Evaluate(
    new EvaluateParameterHandler((s, a) =>
    {
        paramNames.Add(s);
        a.Result = 1; // dummy value
    }),
    new EvaluateFunctionHandler((s, a) =>
    {
        functionNames.Add(s);
        a.Result = 1; // dummy value
    }));