需求: 在C#中對一個字元串進行計算得出結果。 例如:“1+2+66+33” 字元串中的數字可以變化,累加(這裡偷懶了限定成累加)的次數可以變化。 思路/解決方案: 1. 用JavaScript中的Eval把字元串轉成對象進行計算 用Com控制項計算方法,引入MSScriptControl.Scrip ...
需求:
在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); } } } }