原文 http://blog.sunmast.com/sunmast/archive/2004/08/24/870.aspx
/*
* 这是一个简单的CodeDOM示例
* 包括代码动态生成,动态编译,和动态调用
* By Sunmast for wangsaokui
* 2004.8.23
*/
using System;
using System.CodeDom;
using System.CodeDom.Compiler;
using System.IO;
using System.Reflection;
using System.Text;
using Microsoft.CSharp;
namespace Sunmast.Sample.CodeDom
{
class Entry
{
[STAThread]
static void Main(string[] args)
{
// 建立CodeDOM
CodeCompileUnit codeDom = GenerateCodeDom();
// 编译为可用的Assembly
// 这里使用C#代码,实际上任何继承了CodeDomProvider类的Provider都行
// 所以可以换成Microsoft.VisualBasic.VBCodeProvider
Assembly assembly = ComplieCodeDomToAssembly(codeDom,new CSharpCodeProvider());
// 动态调用assembly
if(assembly != null)
{
InvokeMethodsFromAssembly(assembly);
}
Console.ReadLine();
}
static CodeCompileUnit GenerateCodeDom()
{
// 命名空间
CodeNamespace cn = new CodeNamespace("Sunmast.Sample.CodeDom");
cn.Imports.Add(new CodeNamespaceImport("System"));
// 创建新的类型
CodeTypeDeclaration ctd = new CodeTypeDeclaration("SunmastClass");
ctd.IsClass = true;
ctd.TypeAttributes = TypeAttributes.Public;
// 构造函数(并非必须)
CodeConstructor cc = new CodeConstructor();
cc.Attributes = MemberAttributes.Public | MemberAttributes.Final;
ctd.Members.Add(cc);
// 成员函数MyMethod
CodeMemberMethod cmm = new CodeMemberMethod();
cmm.Name = "MyMethod";
cmm.Attributes = MemberAttributes.Public | MemberAttributes.Final;
cmm.Parameters.Add(
new CodeParameterDeclarationExpression(
typeof(string),"input"));
cmm.Statements.Add(
new CodeMethodInvokeExpression(
new CodeSnippetExpression("Console"),"WriteLine",
new CodeArgumentReferenceExpression("input")));
ctd.Members.Add(cmm);
cn.Types.Add(ctd);
// 建立编译单元
CodeCompileUnit unit = new CodeCompileUnit();
unit.Namespaces.Add(cn);
return unit;
}
static Assembly ComplieCodeDomToAssembly(CodeCompileUnit compilationUnit,CodeDomProvider languageProvider)
{
// 预览代码
Console.WriteLine("// 预览代码");
StringBuilder sb = new StringBuilder();
ICodeGenerator icg = languageProvider.CreateGenerator();
icg.GenerateCodeFromCompileUnit(compilationUnit,new StringWriter(sb),null);
Console.WriteLine(sb.ToString());
ICodeCompiler complier = languageProvider.CreateCompiler();
// 建立编译参数
CompilerParameters options = new CompilerParameters(new string[]{"MsCorLib.dll","System.dll"});
// 在内存中建立Assembly
options.GenerateInMemory = true;
// 关闭调式信息,提高性能
options.IncludeDebugInformation = true;
// 忽略警告
options.TreatWarningsAsErrors = false;
// 编译并得到编译结果
CompilerResults results = complier.CompileAssemblyFromDom(options,compilationUnit);
foreach(string str in results.Output)
{
Console.WriteLine(str);
}
if(results.NativeCompilerReturnValue != 0)
{
// 编译失败
Console.WriteLine("!ERROR:");
foreach(CompilerError error in results.Errors)
{
Console.WriteLine(error.ErrorText);
}
return null;
}
else
{
return results.CompiledAssembly;
}
}
static void InvokeMethodsFromAssembly(Assembly assembly)
{
// 得到置于Assembly的class
// 需要写完整命名空间
Type t = assembly.GetType("Sunmast.Sample.CodeDom.SunmastClass",true,true);
// 调用构造函数建立对象
Object obj = t.InvokeMember(null, BindingFlags.CreateInstance, null, null, null);
// Object obj = Activator.CreateInstance(t); // 也可以这么写
// 调用对象的方法(MyMethod)
Console.WriteLine("Invoke Method /"MyMethod/": The END.");
t.InvokeMember("MyMethod",BindingFlags.InvokeMethod, null, obj, new object[]{"The END."});
}
}
}