6

我们有一个使用 Ncalc 计算字符串的通用计算例程。然而,当乘法中的值足够小以至于 Ncalc 将它们视为 int 但结果对于 int 来说太大时,我们遇到了一个问题。

例子:

        var expr = new NCalc.Expression("1740263 * 1234");
        object result = expr.Evaluate();
        Console.WriteLine(result.ToString());
        Console.ReadKey();

这导致负 int 值。

有什么方法可以强制 Ncalc 使用 long 进行计算?

我尝试过使用参数,这很有效,但这意味着要对我们的代码进行重大重写,如果可能的话,我想避免它。

谢谢

4

2 回答 2

4

NCalcInt32用作整数数据类型,那么您不能强制将数字计算为Int64. 但是,如果您不使用内置数学函数并且依赖于普通数学运算符,则可以将数字转换为long,它将调用正确的运算符。让我们来看看它:

var expr = new NCalc.Expression("CLng(1740263) * 1234");
expr.EvaluateFunction += delegate(string name, NCalc.FunctionArgs args)
{
    // Nostalgic CLng function...
    if (String.Equals(name, "CLng", StringComparison.CurrentCultureIgnoreCase))
    {
        args.HasResult = true;
        args.Result = Convert.ToInt64(args.EvaluateParameters()[0]);
    }
};

请注意,您不能直接将 boxedInt32参数转换为,Int64然后您必须使用Convert.ToInt64()或双重转换:(long)(int)args.EvaluateParameters()[0]。现在您可以检查结果是否正确:

var result = expr.Evaluate();
Console.WriteLine("Result is: {0} (type {1})",
    result, result.GetType());

执行正确的类型转换,然后您不需要两个值都转换为long.

请注意,您也可以直接使用浮点数(decimal在 NCalc 中),您不会遇到这样的问题:

var expr = new NCalc.Expression("1740263.0 * 1234.0");
于 2016-03-30T08:35:35.900 回答
0

您可以使用正则表达式从字符串中提取所有数字,将它们替换为参数,然后将参数设置为转换后的数字。在这个例子中,我将 int 数替换为十进制数。

string myCalculationString = "1000000 * 10000000";
striny myPattern = @"-?\d+(,\d+)*(\.\d+(e\d+)?)?";

MatchCollection matches =
Regex.Matches(convertedCalculation, myPattern );

int matchCount = 0;
convertedCalculation = Regex.Replace(convertedCalculation,
myPattern , m => "[p" + matchCount++ + "]");

for (int i = 0; i < matches.Count; i++)
    {
       Match match = matches[i];
       decimal number = Convert.ToDecimal(match.Value);
       expression.Parameters.Add($"p{i}", number);
    }

 decimal result = (decimal)expression.Evaluate();

注意我没有很好地测试正则表达式,但它似乎适用于正数/负数并允许可选的小数点。如果任何数字或结果超出小数范围,这也会引发错误

于 2016-07-01T09:50:58.377 回答