C#實(shí)現(xiàn)動態(tài)執(zhí)行字符串腳本(優(yōu)化版)的示例代碼
像javascript中有eval()來執(zhí)行動態(tài)代碼,c#中是沒有的,于是自己動手豐衣足食,
先來代碼
using System; using System.Data; using System.Configuration; using System.Text; using System.CodeDom.Compiler; using Microsoft.CSharp; using System.Reflection; using System.Collections.Generic; namespace 檢漏儀上位機(jī) { /// <summary> /// 本類用來將字符串轉(zhuǎn)為可執(zhí)行文本并執(zhí)行 /// 從別處復(fù)制,勿隨意更改! /// </summary> public class Evaluator { #region 構(gòu)造函數(shù) /// <summary> /// 可執(zhí)行串的構(gòu)造函數(shù) /// </summary> /// <param name="items"> /// 可執(zhí)行字符串?dāng)?shù)組 /// </param> public Evaluator(EvaluatorItem[] items, Dictionary<string, string> listAssemblies = null) { ConstructEvaluator(items, listAssemblies); //調(diào)用解析字符串構(gòu)造函數(shù)進(jìn)行解析 } /// <summary> /// 可執(zhí)行串的構(gòu)造函數(shù) /// </summary> /// <param name="returnType">返回值類型</param> /// <param name="expression">執(zhí)行表達(dá)式</param> /// <param name="name">執(zhí)行字符串名稱</param> public Evaluator(Type returnType, string expression, string name, Dictionary<string, string> listAssemblies = null) { //創(chuàng)建可執(zhí)行字符串?dāng)?shù)組 EvaluatorItem[] items = { new EvaluatorItem(returnType, expression, name) }; ConstructEvaluator(items, listAssemblies); //調(diào)用解析字符串構(gòu)造函數(shù)進(jìn)行解析 } public Evaluator(string allCode, string _namespace, string _class, List<string> listAssemblies = null) { ConstructEvaluatorByAllCode(allCode, _namespace, _class, listAssemblies); //調(diào)用解析字符串構(gòu)造函數(shù)進(jìn)行解析 } /// <summary> /// 可執(zhí)行串的構(gòu)造函數(shù) /// </summary> /// <param name="item">可執(zhí)行字符串項(xiàng)</param> public Evaluator(EvaluatorItem item, Dictionary<string, string> listAssemblies = null) { EvaluatorItem[] items = { item };//將可執(zhí)行字符串項(xiàng)轉(zhuǎn)為可執(zhí)行字符串項(xiàng)數(shù)組 ConstructEvaluator(items, listAssemblies); //調(diào)用解析字符串構(gòu)造函數(shù)進(jìn)行解析 } /// <summary> /// 解析字符串構(gòu)造函數(shù) /// </summary> /// <param name="items">待解析字符串?dāng)?shù)組</param> private void ConstructEvaluator(EvaluatorItem[] items, Dictionary<string, string> listAssemblies = null) { //創(chuàng)建C#編譯器實(shí)例 ICodeCompiler comp = (new CSharpCodeProvider().CreateCompiler()); //編譯器的傳入?yún)?shù) CompilerParameters cp = new CompilerParameters(); cp.ReferencedAssemblies.Add("system.dll"); //添加程序集 system.dll 的引用 cp.ReferencedAssemblies.Add("system.data.dll"); //添加程序集 system.data.dll 的引用 cp.ReferencedAssemblies.Add("system.xml.dll"); //添加程序集 system.xml.dll 的引用 cp.ReferencedAssemblies.Add("System.Windows.Forms.dll"); //添加程序集 system.xml.dll 的引用 cp.GenerateExecutable = false; //不生成可執(zhí)行文件 cp.GenerateInMemory = true; //在內(nèi)存中運(yùn)行 StringBuilder code = new StringBuilder(); //創(chuàng)建代碼串 if (listAssemblies != null) { foreach (var item in listAssemblies) { cp.ReferencedAssemblies.Add(item.Key); //添加程序集 引用 code.Append("using "+item.Value + "; \n"); } } /* * 添加常見且必須的引用字符串 */ code.Append("using System; \n"); code.Append("using System.Data; \n"); code.Append("using System.Data.SqlClient; \n"); code.Append("using System.Data.OleDb; \n"); code.Append("using System.Xml; \n"); code.Append("using System.Windows.Forms; \n"); code.Append("namespace EvalGuy { \n"); //生成代碼的命名空間為EvalGuy,和本代碼一樣 code.Append(" public class _Evaluator { \n"); //產(chǎn)生 _Evaluator 類,所有可執(zhí)行代碼均在此類中運(yùn)行 foreach (EvaluatorItem item in items) //遍歷每一個可執(zhí)行字符串項(xiàng) { code.AppendFormat(" public {0} {1}() ", //添加定義公共函數(shù)代碼 item.ReturnType == null ? "void" : item.ReturnType.Name, //函數(shù)返回值為可執(zhí)行字符串項(xiàng)中定義的返回值類型 item.Name); //函數(shù)名稱為可執(zhí)行字符串項(xiàng)中定義的執(zhí)行字符串名稱 code.Append("{ "); //添加函數(shù)開始括號 if (item.ReturnType != null) code.AppendFormat("return ({0});", item.Expression);//添加函數(shù)體,返回可執(zhí)行字符串項(xiàng)中定義的表達(dá)式的值 else { code.Append(item.Expression);//添加函數(shù)體,返回可執(zhí)行字符串項(xiàng)中定義的表達(dá)式的值 } code.Append("}\n"); //添加函數(shù)結(jié)束括號 } code.Append("} }"); //添加類結(jié)束和命名空間結(jié)束括號 //得到編譯器實(shí)例的返回結(jié)果 CompilerResults cr = comp.CompileAssemblyFromSource(cp, code.ToString()); if (cr.Errors.HasErrors) //如果有錯誤 { StringBuilder error = new StringBuilder(); //創(chuàng)建錯誤信息字符串 error.Append("編譯有錯誤的表達(dá)式: "); //添加錯誤文本 foreach (CompilerError err in cr.Errors) //遍歷每一個出現(xiàn)的編譯錯誤 { error.AppendFormat("{0}\n", err.ErrorText); //添加進(jìn)錯誤文本,每個錯誤后換行 } throw new Exception("編譯錯誤: " + error.ToString());//拋出異常 } Assembly a = cr.CompiledAssembly; //獲取編譯器實(shí)例的程序集 _Compiled = a.CreateInstance("EvalGuy._Evaluator"); //通過程序集查找并聲明 EvalGuy._Evaluator 的實(shí)例 } private void ConstructEvaluatorByAllCode(string allcode, string _namespace, string _class, List<string> listAssemblies) { if (listAssemblies == null) { listAssemblies = new List<string>(); } //創(chuàng)建C#編譯器實(shí)例 ICodeCompiler comp = (new CSharpCodeProvider().CreateCompiler()); //編譯器的傳入?yún)?shù) CompilerParameters cp = new CompilerParameters(); if (!listAssemblies.Contains("system.dll")) { listAssemblies.Add("system.dll"); listAssemblies.Add("system.data.dll"); listAssemblies.Add("system.xml.dll"); } foreach (var item in listAssemblies) { cp.ReferencedAssemblies.Add(item); } cp.GenerateExecutable = false; //不生成可執(zhí)行文件 cp.GenerateInMemory = true; //在內(nèi)存中運(yùn)行 //得到編譯器實(shí)例的返回結(jié)果 CompilerResults cr = comp.CompileAssemblyFromSource(cp, allcode); if (cr.Errors.HasErrors) //如果有錯誤 { StringBuilder error = new StringBuilder(); //創(chuàng)建錯誤信息字符串 error.Append("編譯有錯誤的表達(dá)式: "); //添加錯誤文本 foreach (CompilerError err in cr.Errors) //遍歷每一個出現(xiàn)的編譯錯誤 { error.AppendFormat("{0}\n", err.ErrorText); //添加進(jìn)錯誤文本,每個錯誤后換行 } throw new Exception("編譯錯誤: " + error.ToString());//拋出異常 } Assembly a = cr.CompiledAssembly; //獲取編譯器實(shí)例的程序集 _Compiled = a.CreateInstance($"{_namespace}.{_class}"); //通過程序集查找并聲明 EvalGuy._Evaluator 的實(shí)例 } #endregion #region 公有成員 /// <summary> /// 執(zhí)行字符串并返回整型值 /// </summary> /// <param name="name">執(zhí)行字符串名稱</param> /// <returns>執(zhí)行結(jié)果</returns> public int EvaluateInt(string name) { return (int)Evaluate(name); } /// <summary> /// 執(zhí)行字符串并返回字符串型值 /// </summary> /// <param name="name">執(zhí)行字符串名稱</param> /// <returns>執(zhí)行結(jié)果</returns> public string EvaluateString(string name) { return (string)Evaluate(name); } /// <summary> /// 執(zhí)行字符串并返回布爾型值 /// </summary> /// <param name="name">執(zhí)行字符串名稱</param> /// <returns>執(zhí)行結(jié)果</returns> public bool EvaluateBool(string name) { return (bool)Evaluate(name); } /// <summary> /// 執(zhí)行字符串并返 object 型值 /// </summary> /// <param name="name">執(zhí)行字符串名稱</param> /// <returns>執(zhí)行結(jié)果</returns> public object Evaluate(string name) { MethodInfo mi = _Compiled.GetType().GetMethod(name);//獲取 _Compiled 所屬類型中名稱為 name 的方法的引用 return mi.Invoke(_Compiled, null); //執(zhí)行 mi 所引用的方法 } #endregion #region 靜態(tài)成員 /// <summary> /// 執(zhí)行表達(dá)式并返回整型值 /// </summary> /// <param name="code">要執(zhí)行的表達(dá)式</param> /// <param name="listAssemblies">需要引用到類庫,key:dll名稱,value:命名空間名稱</param> /// <returns>運(yùn)算結(jié)果</returns> static public int EvaluateToInteger(string code, Dictionary<string, string> listAssemblies = null) { Evaluator eval = new Evaluator(typeof(int), code, staticMethodName, listAssemblies);//生成 Evaluator 類的對像 return (int)eval.Evaluate(staticMethodName); //執(zhí)行并返回整型數(shù)據(jù) } /// <summary> /// 執(zhí)行表達(dá)式并返回字符串型值 /// </summary> /// <param name="code">要執(zhí)行的表達(dá)式</param> /// <param name="listAssemblies">需要引用到類庫,key:dll名稱,value:命名空間名稱</param> /// <returns>運(yùn)算結(jié)果</returns> static public string EvaluateToString(string code, Dictionary<string, string> listAssemblies = null) { Evaluator eval = new Evaluator(typeof(string), code, staticMethodName, listAssemblies);//生成 Evaluator 類的對像 return (string)eval.Evaluate(staticMethodName); //執(zhí)行并返回字符串型數(shù)據(jù) } /// <summary> /// 執(zhí)行表達(dá)式并返回布爾型值 /// </summary> /// <param name="code">要執(zhí)行的表達(dá)式</param> /// <param name="listAssemblies">需要引用到類庫,key:dll名稱,value:命名空間名稱</param> /// <returns>運(yùn)算結(jié)果</returns> static public bool EvaluateToBool(string code, Dictionary<string, string> listAssemblies = null) { Evaluator eval = new Evaluator(typeof(bool), code, staticMethodName, listAssemblies);//生成 Evaluator 類的對像 return (bool)eval.Evaluate(staticMethodName); //執(zhí)行并返回布爾型數(shù)據(jù) } /// <summary> /// 執(zhí)行表達(dá)式并返回 object 型值 /// </summary> /// <param name="code">要執(zhí)行的表達(dá)式</param> /// <param name="listAssemblies">需要引用到類庫,key:dll名稱,value:命名空間名稱</param> /// <returns>運(yùn)算結(jié)果</returns> static public object EvaluateToObject(string code, Dictionary<string, string> listAssemblies = null) { Evaluator eval = new Evaluator(typeof(object), code, staticMethodName, listAssemblies);//生成 Evaluator 類的對像 return eval.Evaluate(staticMethodName); //執(zhí)行并返回 object 型數(shù)據(jù) } /// <summary> /// 執(zhí)行一個無返回式的代碼 /// </summary> /// <param name="code"></param> /// <param name="listAssemblies">需要引用到類庫,key:dll名稱,value:命名空間名稱</param> static public void EvaluateToVoid(string code, Dictionary<string, string> listAssemblies = null) { Evaluator eval = new Evaluator(null, code, staticMethodName, listAssemblies);//生成 Evaluator 類的對像 eval.Evaluate(staticMethodName); //執(zhí)行并返回 object 型數(shù)據(jù) } /// <summary> /// 執(zhí)行一個全代碼 /// </summary> /// <param name="code">全代碼,包含命名空間引用,命名空間聲明,類聲明,函數(shù)聲明等</param> /// <param name="_namespace"></param> /// <param name="_class"></param> /// <param name="methodName">函數(shù)</param> /// <param name="listAssemblies">需要引用到類庫</param> public static void EvaluateByAllCode(string code, string _namespace, string _class, string methodName, List<string> listAssemblies = null) { Evaluator eval = new Evaluator(code, _namespace, _class, listAssemblies);//生成 Evaluator 類的對像 eval.Evaluate(methodName); } #endregion #region 私有成員 /// <summary> /// 靜態(tài)方法的執(zhí)行字符串名稱 /// </summary> private const string staticMethodName = "__foo"; /// <summary> /// 用于動態(tài)引用生成的類,執(zhí)行其內(nèi)部包含的可執(zhí)行字符串 /// </summary> object _Compiled = null; #endregion } /// <summary> /// 可執(zhí)行字符串項(xiàng)(即一條可執(zhí)行字符串) /// </summary> public class EvaluatorItem { /// <summary> /// 返回值類型 /// </summary> public Type ReturnType; /// <summary> /// 執(zhí)行表達(dá)式 /// </summary> public string Expression; /// <summary> /// 執(zhí)行字符串名稱 /// </summary> public string Name; /// <summary> /// 可執(zhí)行字符串項(xiàng)構(gòu)造函數(shù) /// </summary> /// <param name="returnType">返回值類型</param> /// <param name="expression">執(zhí)行表達(dá)式</param> /// <param name="name">執(zhí)行字符串名稱</param> public EvaluatorItem(Type returnType, string expression, string name) { ReturnType = returnType; Expression = expression; Name = name; } } }
調(diào)用一個無返回值的代碼,顯示一個提示框
Evaluator.EvaluateToVoid("MessageBox.Show(\"Test\");",new Dictionary<string, string>() { { "System.Windows.Forms.dll", "System.Windows.Forms" } });
調(diào)用一個計(jì)算返回整型
Evaluator.EvaluateToInteger("1+2*3");
調(diào)用一個全代碼
string str = @"using System; namespace a { public class b { public void c() { Console.WriteLine(1); } } }"; Evaluator.EvaluateByAllCode(str, "a", "b", "c");
功能就這么多
以上就是C#實(shí)現(xiàn)動態(tài)執(zhí)行字符串腳本(優(yōu)化版)的示例代碼的詳細(xì)內(nèi)容,更多關(guān)于C#動態(tài)執(zhí)行字符串腳本的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
程序中兩個Double類型相加出現(xiàn)誤差的解決辦法
本篇文章介紹了,程序中兩個Double類型相加出現(xiàn)誤差的解決辦法。需要的朋友參考下2013-04-04C#調(diào)用SQL?Server中有參數(shù)的存儲過程
這篇文章介紹了C#調(diào)用SQL?Server中有參數(shù)存儲過程的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-03-03一文詳解C#中數(shù)組、鏈表、Hash的優(yōu)缺點(diǎn)
在 C# 中,數(shù)組(Array)、鏈表(LinkedList)和哈希表(Hash)是常用的數(shù)據(jù)結(jié)構(gòu),每種都有其自身的優(yōu)缺點(diǎn),本文將通過代碼示例給大家詳細(xì)的介紹一下,需要的朋友可以參考下2024-02-02C#中Invoke和BeginInvoke實(shí)際應(yīng)用詳解
這篇文章主要給大家介紹了關(guān)于C#中Invoke和BeginInvoke實(shí)際應(yīng)用的相關(guān)資料,Invoke是對象方法,BeginInvoke是靜態(tài)方法,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-12-12C#實(shí)現(xiàn)應(yīng)用程序的監(jiān)控與調(diào)試的示例代碼
日志記錄是軟件開發(fā)中不可或缺的功能,它能幫助開發(fā)者在應(yīng)用程序運(yùn)行時記錄重要信息,本文就來介紹一下常用日志記錄功能以及常用的日志庫,感興趣的可以了解一下2024-03-03C#使用百度Ueditor富文本框?qū)崿F(xiàn)上傳文件
這篇文章主要為大家詳細(xì)介紹了C#如何使用百度Ueditor富文本框?qū)崿F(xiàn)上傳文件(圖片,視頻等),文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-07-07