计算字符串得出结果

本文介绍在C#中使用多种方法对字符串表达式进行计算,包括使用JavaScript的Eval函数、DataTable.Compute方法、SQL查询及表达式树。特别深入探讨了如何利用表达式树动态创建累加表达式,并提供了完整的代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

需求:

在C#中对一个字符串进行计算得出结果。

例如:“1+2+66+33”

字符串中的数字可以变化,累加(这里偷懒了限定成累加)的次数可以变化。

思路/解决方案:

1. 用JavaScript中的Eval把字符串转成对象进行计算

用Com控件计算方法,引入MSScriptControl.ScriptControlClass

2.用DataTable().Compute计算方法

3.把字符串表达式作为一个sql发到数据库计算后结果再回传到程序

4.分解字符串用 lambda表达式树动态创建一个(累加)表达式

 

说明:

我做了一个随机数,然后将每次随机出来的数字拼接成一个字符串展示出来。

再来解析这个字符串得出计算结果。

需要注意表达式树的方式,你的参数列表paramslist必须是  object[] 这个类型,我之前用的int[] 也可以编译但是在运行的时候会报错,你的paramenterArray应当就是构成binaryexpression的那些ParameterExpression。

如果你的业务里面要用到更加复杂运算,你需要亲自来尝试。

欢迎吐槽。

 

代码:

using System;
using System.Collections.Generic;
using System.Text;
using System.Linq.Expressions;
using System.Reflection;

namespace guess
{
    class Program
    {
        static void Main(string[] args)
        {
            try
            {
                Console.WriteLine(SetRandomVariable(GetTemplate()));
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
            }        

#if DEBUG
            Console.Read();
#endif


        }

        private static string GetTemplate()
        {

            StringBuilder stb = new StringBuilder();
            stb.AppendLine("Bangladesh - OneERP {0}hr");
            stb.AppendLine("CN - OneERP {1}hr");
            stb.AppendLine("Global - OneERP {2}hr");
            stb.AppendLine("FE - OneERP {3}hr");
            stb.AppendLine("UK - OneERP {4}hr");
            stb.AppendLine("Turkey - OneERP {5}hr");
            stb.AppendLine("Vietnam – OneERP {6}hr");
            stb.AppendLine("Singapore - OneERP {7}hr");
            stb.AppendLine("CN - HZ {8}hr");
            stb.AppendLine("Romania - OneERP {9}hr");
            stb.AppendLine("Thailand - OneERP {10}hr");
            stb.AppendLine("ZheJiang - OneERP {11}hr");
            stb.AppendLine("Global - OneERP SSO {12}hr");

            return stb.ToString();

        }

        private static string SetRandomVariable(string getTemplate)
        {

            int count = getTemplate.Split('\n').Length - 1;

            Random randomVariable = new Random(DateTime.Now.Millisecond);
            int max = 40;
          
            StringBuilder formula = new StringBuilder();
            object[] paramslist = new object[count];
            for (int i = 0; i < paramslist.Length; i++)
            {
                if (i == paramslist.Length - 1)
                {
                    paramslist[i] = max;
                    formula.Append(max);
                }
                else
                {
                    paramslist[i] = available(ref max, ref formula);
                }

            }

            //Com控件计算方法 需要目标平台X86 D找到引用的dll——属性——嵌入互操作类型——false
            MSScriptControl.ScriptControl sc = new MSScriptControl.ScriptControlClass();
            sc.Language = "JavaScript";
            Console.WriteLine("{0}={1}", formula.ToString(), sc.Eval(formula.ToString()).ToString());


            //DataTable().Compute计算方法
            Console.WriteLine("{0}={1}", formula.ToString(), new System.Data.DataTable().Compute(formula.ToString(), "").ToString());

            //另外有一种思路,把表达式作为一个sql发到数据库计算后结果再回传到程序。

            //lambda表达式树动态创建一个累加的表达式 需要.net FrameWork 4.5
            BinaryExpression binaryexpression = addExpression(count,null);
            Console.WriteLine(binaryexpression);
            LambdaExpression lambdaExpr = Expression.Lambda(binaryexpression, paramenterArray);
            Delegate de = lambdaExpr.Compile();
           // MethodInfo method = de.Method;
            Console.WriteLine(de.DynamicInvoke(paramslist));
            return string.Format(getTemplate, paramslist);
           
        }

        private static int available(ref int max, ref StringBuilder formula)
        {


            int temp = 0;
            if (max > 0)
            {
                Random randomVariable = new Random();
                if (max == 1)
                {
                    temp = randomVariable.Next(0, max + 1);
                }
                else
                {
                    temp = randomVariable.Next(0, max);
                }
                max = max - temp;
            }
            formula.AppendFormat("{0}+", temp);
            return temp;

        }
       static List<ParameterExpression>  paramenterArray=new List<ParameterExpression>() ;

        /// <summary>
        /// 
        /// </summary>
        /// <param name="lenght">累加次数</param>
        /// <param name="resultLeft">默认累加表达式</param>
        /// <returns></returns>
        private static BinaryExpression addExpression(int lenght, BinaryExpression resultLeft)
        {

            try
            {
                ParameterExpression left, right;
                left = Expression.Parameter(typeof(int), string.Format("left{0}", lenght) );
                right = Expression.Parameter(typeof(int), string.Format("right{0}", (lenght - 1)) );
                BinaryExpression result = Expression.Add(left, right);

                if (lenght > 2)
                {
                    paramenterArray.Add(left);
                    paramenterArray.Add(right);
                    if (resultLeft != null)
                        result = Expression.Add(resultLeft, addExpression(lenght - 2, result));
                    else
                        result = addExpression(lenght - 2, result);

                }
                if (lenght == 2)
                {
                    paramenterArray.Add(left);
                    paramenterArray.Add(right);
                    if (resultLeft != null)
                        result = Expression.Add(resultLeft, result);
                    else
                    {
                        //ignore;
                    }
                }
                if (lenght == 1)
                {
                    if (resultLeft != null)
                    {
                        ParameterExpression rExpression = Expression.Parameter(typeof(int), string.Format("right{0}", lenght ));
                        paramenterArray.Add(rExpression);
                        result = Expression.Add(resultLeft, rExpression);
                    }
                    else
                    {

                        throw (new Exception("当lenght等于1时,resultLeft参数不能为空"));
                    }
                }
                return result;
            }
            catch ( Exception ex)
            {
#if DEBUG
                Console.WriteLine(ex.StackTrace.ToString());
#endif
                throw (ex);
            }
        }
    }
}

 收录羽铉君的四则混合运算写法。Pop(), Push(opt) ,Peek()的范例。

    public class Calculator
    {
        public Calculator()
        {
            _OptStack = new Stack<char>();
            _SuffixStack = new Stack<float>();
        }

        private Stack<char> _OptStack;
        private Stack<float> _SuffixStack;

        public float Calculate(string expression)
        {
            string lastNum = string.Empty;
            for (int i = 0; i < expression.Length; i++)
            {
                if (char.IsNumber(expression[i]) || expression[i].Equals('.'))
                {
                    lastNum += expression[i];
                }
                else
                {
                    if (lastNum != string.Empty)
                    {
                        Merger(float.Parse(lastNum));
                        lastNum = string.Empty;
                    }
                    AddOpt(expression[i]);
                }
            }
            if (lastNum != string.Empty)
            {
                Merger(float.Parse(lastNum));
            }
            while (_OptStack.Count > 0)
            {
                Merger(_OptStack.Pop());
            }

            return _SuffixStack.Pop();
        }

        private void AddOpt(char opt)
        {
            if (_OptStack.Count == 0)
            {
                _OptStack.Push(opt);
                return;
            }
            if (opt.Equals(')'))
            {
                while (!_OptStack.Peek().Equals('('))
                {
                    Merger(_OptStack.Pop());
                }
                _OptStack.Pop();
                return;
            }
            char tempOpt = _OptStack.Peek();
            if ((opt.Equals('-') || opt.Equals('+')) &&
            (tempOpt.Equals('*') || tempOpt.Equals('/')))
            {
                while (_OptStack.Count > 0)
                {
                    Merger(_OptStack.Pop());
                }
            }

            _OptStack.Push(opt);
        }

        private void Merger(float exp)
        {
            _SuffixStack.Push(exp);
        }

        private void Merger(char exp)
        {
            float num1 = _SuffixStack.Pop();
            float num2 = _SuffixStack.Pop();
            float result = 0;
            switch (exp)
            {
                case '+':
                    result = num2 + num1;
                    break;
                case '-':
                    result = num2 - num1;
                    break;
                case '*':
                    result = num2 * num1;
                    break;
                case '/':
                    result = num2 / num1;
                    break;
            }
            _SuffixStack.Push(result);
        }
    }

    //调用写法
    //string calc = "89+(126/2)*3-6/3+(96+3-8/2)";// 随便写四则运算
    //Calculator calculator = new Calculator();
    //MessageBox.Show(calculator.Calculate(calc).ToString()); // 结果

 

转载于:https://www.cnblogs.com/mlfg/p/10411603.html

因各个项目中需要使用根据字符串计算数值,这里写出一个算法,专门计算字符串。配有大量常用公式。只有一个人方法,直接调用即可。 类名:CustomMath 函数名:Calculations(string value) 说明:求解算式表达式字符串的值 表达式中包含的符号或函数: truncate, ceiling,floor,round,log10, sign,sinh,sqrt, asin,atan,cosh, tanh, sin,cos,tan ,abs,acos, exp,log,max,min,pow,mod,+,-,*,/,',',(,) 函数说明:(不区分大小写) truncate(num) 计算指定数的整数部分 truncate(1.23)=1 ceiling (num) 返回大于或等于指定的双精度浮点数的最小整数值 ceiling(1.23)=2 floor(num) 返回小于或等于指定双精度浮点数的最大整数 floor(1.23)=1 round(num) 将双精度浮点值舍入为最接近的整数值 round(1.23)=1 round(num,num1) 将小数值按指定的小数位数舍入 round(1.23,1)=1.2 log10(num) 返回指定数字以 10 为底的对数 log10(10)=1 sign(num) 返回表示数字符号的值 sign(1.23)=1 sinh(num) 返回指定角度的双曲正弦值 sinh(1.23)=1.5644 sqrt(num) 返回指定数字的平方根 sqrt(9)=3 sqrt(num,num1) 返回指定数字的num1根 sqrt(27,3)=3 asin(num) 返回正弦值为指定数字的角度 asin(0.5)=PI/6 atan(num) 返回正切值为指定数字的角度 atan(1)=45 cosh(num) 返回指定角度的双曲余弦值 cosh(1.23)=1.8567 tanh(num) 返回指定角度的双曲正切值 tanh(1.23)=0.8425 sin(num) 返回指定角度的正弦值 sin(PI/6)=0.5 cos(num) 返回指定角度的余弦值 sin(PI/3)=0.5 tan(num) 返回指定角度的余切值 sin(PI/4)=1 abs(num) 返回数字的绝对值 abs(-12)=12 acos(num) 返回余弦值为指定数字的角度 acos(0.5)=PI/3 exp(num) 返回 e 的指定次幂 exp(1)=2.718 log(num) 返回指定数字的自然对数(底为 e) log(e)=1 log(num,num1) 返回指定数字在使用指定底时的对数 log(e,e)=1 max(num,um1) 返回最大值 max(1,2)=2 min(num,num1) 返回最小值 min(1,2)=1 pow(num,num1) 返回指定数字的指定次幂 pow(2,2)=4 mod(num,num1) 返回余数 mod(3,2)=1 常量: PI 值:3.14159265358979323846 E 值:2.7182818284590452354 YEAR 值:当前年份 MONTH 值:当前月份 DAY 值: 当前日 HOUR 值:当前时 MINUTE 值:当前分 SECOND 值:当前秒 RANDOM 值:一个随机数(0-1 之间) 实例 系统计算:1+2*3/4-0.5=2 函数计算:1+2*3/4-0.5=2 调用方式:CustomMath.Calculations("1+2*3/4-0.5") 系统计算:(1+2)*3/4-0.5=1.75 函数计算:(1+2)*3/4-0.5=1.75 调用方式:CustomMath.Calculations("(1+2)*3/4-0.5") 系统计算:(sin(pi)+sqrt(3+5*7+(2+8/4*5+2)))/6=1.20185042515466 公式计算:(sin(pi)+sqrt(3+5*7+(2+8/4*5+2)))/6=1.20185042515466 调用方式:CustomMath.Calculations("(sin(pi)+sqrt(3+5*7+(2+8/4*5+2)))/6") 系统计算:sin(pow(3,2)/4)+3.5-9*sqrt(81)=-76.7219268031121 函数计算:sin(pow(3,2)/4)+3.5-9*sqrt(81)=-76.7219268031121 调用方式:CustomMath.Calculations("sin(pow(3,2)/4)+3.5-9*sqrt(81)")
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值