代码的动态编译并执行是一个NET平台提供给我们的很强大的工具用以灵活扩展(当然是面对内部开发人员)复杂而无法估算的逻辑并通过一些额外的代码来扩展我们已有 的应用程序这在很大程度上给我们提供了另外一种扩展的方式(当然这并不能算是严格意义上的扩展但至少为我们提供了一种思路)
动态代码执行可以应用在诸如模板生成外加逻辑扩展等一些场合一个简单的例子为了网站那的响应速度HTML静态页面往往是我们最好的选择但基于数据驱动的网站往往又很难用静态页面实现那么将动态页面生成html的工作或许就是一个很好的应用场合另外对于一些模板的套用我们同样可以用它来做另外这本身也是插件编写的方式
最基本的动态编译
Net为我们提供了很强大的支持来实现这一切我们可以去做的基础主要应用的两个命名空间是SystemCodeDomCompiler和MicrosoftCSharp或MicrosoftVisualBasic另外还需要用到反射来动态执行你的代码动态编译并执行代码的原理其实在于将提供的源代码交予CSharpCodeProvider来执行编译(其实和CSC没什么两样)如果没有任何编译错误生成的IL代码会被编译成DLL存放于于内存并加载在某个应用程序域(默认为当前)内并通过反射的方式来调用其某个方法或者触发某个事件等之所以说它是插件编写的一种方式也正是因为与此我们可以通过预先定义好的借口来组织和扩展我们的程序并将其交还给主程序去触发一个基本的动态编译并执行代码的步骤包括
· 将要被编译和执行的代码读入并以字符串方式保存
· 声明CSharpCodeProvider对象实例
· 调用CSharpCodeProvider实例的CompileAssemblyFromSource方法编译
· 用反射生成被生成对象的实例(AssemblyCreateInstance)
· 调用其方法
以下代码片段包含了完整的编译和执行过程
//get the code to compilestring strSourceCode = thistxtSourceText;// Create a new CSharpCodePrivoder instanceCSharpCodeProvider objCSharpCodePrivoder = new CSharpCodeProvider();// Sets the runtime compiling parameters
by crating a new CompilerParameters instanceCompilerParameters objCompilerParameters = new CompilerParameters();objCompilerParametersReferencedAssembliesAdd(Systemdll);objCompilerParametersReferencedAssembliesAdd(SystemWindowsFormsdll);objCompilerParametersGenerateInMemory = true;// CompilerResults: Complile the code snippet
by calling a method from the providerCompilerResults cr = objCSharpCodePrivoder
CompileAssemblyFromSource(objCompilerParameters
strSourceCode);if (crErrorsHasErrors){ string strErrorMsg = crErrorsCountToString() + Errors:; for (int x = ; x < crErrorsCount; x++) { strErrorMsg = strErrorMsg + \r\nLine: + crErrors[x]LineToString() + + crErrors[x]ErrorText; } thistxtResultText = strErrorMsg; MessageBoxShow(There were build erros please modify your code
Compiling Error); return;}// Invoke the method by using ReflectionAssembly objAssembly = crCompiledAssembly;object objClass = objAssemblyCreateInstance(DynamiclyHelloWorld);if (objClass == null){ thistxtResultText = Error: + Couldnt load class; return;}object[] objCodeParms = new object[];objCodeParms[] = Allan;string strResult = (string)objClassGetType()InvokeMember( GetTime BindingFlagsInvokeMethod null objClass objCodeParms); thistxtResultText = strResult;
需要解释的是这里我们在传递编译参数时设置了GenerateInMemory为true这表明生成的DLL会被加载在内存中(随后被默认引用入当前应用程序域)在调用GetTime方法时我们需要加入参数传递object类型的数组并通过Reflection的InvokeMember来调用在创建生成的Assembly中的对象实例时需要注意用到的命名空间是你输入代码的真实命名空间以下是我们输入的测试代码(为了方便所有的代码都在外部输入动态执行时不做调整)
using System;namespace Dynamicly{ public class HelloWorld { public string GetTime(string strName) { return Welcome + strName +
Check in at + SystemDateTimeNowToString(); } } }
运行附件中提供的程序可以很容易得到一下结果
)thisstylewidth=; border= twffan=done>
改进的执行过程
现在一切看起来很好我们可以编译代码并把代码加载到当前应用程序域中来参与我们的活动但你是否想过去卸载掉这段程序呢?更好的去控制程序呢?另外当你运行这个程序很多遍的时候你会发现占用内存很大而且每次执行都会增大内存使用是否需要来解决这个问题呢?当然需要否则你会发现这个东西根本没用我需要执行的一些大的应用会让我的服务器crzay不堪重负而疯掉的
要解决这个问题我们需要来了解一下应用程序域NET Application Domain是NET提供的运行和承载一个活动的进程(Process)的容器它将这个进程运行所需的代码和数据隔离到一个小的范围内称为Application Domain当一个应用程序运行时Application Domains将所有的程序集/组件集加载到当前的应用程序域中并根据需要来调用而对于动态生成的代码/程序集我们看起来好像并没有办法去管理它其实不然我们可以用Application Domain提供的管理程序集的办法来动态加载和移除Assemblies来达到我们的提高性能的目的具体怎么做呢在前边的基础上增加以下步骤
· 创建另外一个Application Domain
· 动态创建(编译)代码并保存到磁盘
· 创建一个公共的远程调用接口
· 创建远程调用接口的实例并通过这个接口来访问其方法
换句话来讲就是将对象加载到另外一个AppDomain中并通过远程调用的方法来调用所谓远程调用其实也就是跨应用程序域调用所以这个对象(动态代码)必须继承于MarshalByRefObject类为了复用这个接口被单独提到一个工程中并提供一个工厂来简化每次的调用操作
borderColorDark=#ffffff cellPadding= width= align=center borderColorLight=# border=>ee>
using System;using SystemCollectionsGeneric;using SystemLinq;using SystemText;using SystemReflection;namespace RemoteAccess{ /// /// Interface that can be run over the remote AppDomain boundary /// public interface IRemoteInterface { object Invoke(string lcMethodobject[] Parameters); } /// /// Factory class to create objects exposing IRemoteInterface /// public class RemoteLoaderFactory : MarshalByRefObject { private const BindingFlags bfi = BindingFlags
Instance | BindingFlagsPublic
| BindingFlagsCreateInstance; public RemoteLoaderFactory() {} public IRemoteInterface Create( string assemblyFile string typeName
object[] constructArgs ) { return (IRemoteInterface) ActivatorCreateInstanceFrom( assemblyFile typeName false bfi null constructArgs null null null )Unwrap(); } } }
接下来在原来基础上需要修改的是
· 将编译成的DLL保存到磁盘中
· 创建另外的AppDomain
· 获得IRemoteInterface接口的引用
(将生成的DLL加载到额外的AppDomain)
· 调用InvokeMethod方法来远程调用
· 可以通过AppDomainUnload()方法卸载程序集
以下是完整的代码演示了如何应用这一方案
borderColorDark=#ffffff cellPadding= width= align=center borderColorLight=# border=>ee>
//get the code to compilestring strSourceCode = thistxtSourceText;// Create an addtional AppDomainAppDomainSetup objSetup = new AppDomainSetup();objSetupApplicationBase = AppDomainCurrentDomainBaseDirectory;AppDomain objAppDomain = AppDomainCreateDomain(MyAppDomain null objSetup);// Create a new CSharpCodePrivoder instanceCSharpCodeProvider objCSharpCodePrivoder = new CSharpCodeProvider();// Sets the runtime compiling parameters
by crating a new CompilerParameters instanceCompilerParameters objCompilerParameters = new CompilerParameters();objCompilerParametersReferencedAssembliesAdd(Systemdll);objCompilerParametersReferencedAssembliesAdd(SystemWindowsFormsdll);// Load the remote loader interfaceobjCompilerParametersReferencedAssembliesAdd(RemoteAccessdll);// Load the resulting assembly into memoryobjCompilerParametersGenerateInMemory = false;objCompilerParametersOutputAssembly = DynamicalCodedll;// CompilerResults: Complile the code snippet
by calling a method from the providerCompilerResults cr = objCSharpCodePrivoder
CompileAssemblyFromSource(objCompilerParameters strSourceCode);if (crErrorsHasErrors){ string strErrorMsg = crErrorsCountToString() + Errors:; for (int x = ; x < crErrorsCount; x++) { strErrorMsg = strErrorMsg + \r\nLine: + crErrors[x]LineToString() + + crErrors[x]ErrorText; } thistxtResultText = strErrorMsg; MessageBoxShow(There were build erros
please modify your code Compiling Error); return;}// Invoke the method by using ReflectionRemoteLoaderFactory factory = (RemoteLoaderFactory)objAppDomain
CreateInstance(RemoteAccessRemoteAccessRemoteLoaderFactory)Unwrap();// with help of factory create a real LiveClass instanceobject objObject = factoryCreate(DynamicalCodedll
DynamiclyHelloWorld null);if (objObject == null){ thistxtResultText = Error: + Couldnt load class; return;}// *** Cast object to remote interface avoid loading type infoIRemoteInterface objRemote = (IRemoteInterface)objObject;object[] objCodeParms = new object[];objCodeParms[] = Allan;string strResult = (string)objRemoteInvoke(GetTime objCodeParms);thistxtResultText = strResult;//Dispose the objects and unload the generated DLLsobjRemote = null;AppDomainUnload(objAppDomain); SystemIOFileDelete(DynamicalCodedll);
对于客户端的输入程序我们需要继承于MarshalByRefObject类和IRemoteInterface接口并添加对RemoteAccess程序集的引用以下为输入
borderColorDark=#ffffff cellPadding= width= align=center borderColorLight=# border=>ee>
using System;using SystemReflection;using RemoteAccess;namespace Dynamicly{ public class HelloWorld : MarshalByRefObjectIRemoteInterface { public object Invoke(string strMethodobject[] Parameters) { return thisGetType()InvokeMember(strMethod BindingFlags
InvokeMethodnullthisParameters); } public string GetTime(string strName) { return Welcome + strName +
Check in at + SystemDateTimeNowToString(); } }}
这样你可以通过适时的编译加载和卸载程序集来保证你的程序始终处于一个可控消耗的过程并且达到了动态编译的目的而且因为在不同的应用程序域中让你的本身的程序更加安全和健壮