1. 程式人生 > >秒懂C#通過Emit動態生成代碼

秒懂C#通過Emit動態生成代碼

動態生成 defined 卸載 fun 磁盤 sse C# loki names

首先需要聲明一個程序集名稱,

1 // specify a new assembly name
2 var assemblyName = new AssemblyName("Kitty");

從當前應用程序域獲取程序集構造器,

1 // create assembly builder
2 var assemblyBuilder = AppDomain.CurrentDomain
3   .DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);

有幾種動態程序集構造訪問限制:

  • AssemblyBuilderAccess.Run; 表示程序集可被執行,但不能被保存。  
  • AssemblyBuilderAccess.Save; 表示程序集可被保存,但不能被執行。  
  • AssemblyBuilderAccess.RunAndSave; 表示程序集可被保存並能被執行。
  • AssemblyBuilderAccess.ReflectionOnly; 表示程序集只能用於反射上下文環境中,不能被執行。 
  • AssemblyBuilderAccess.RunAndCollect; 表示程序集可以被卸載並且內存會被回收。

在程序集中構造動態模塊,

1 // create module builder
2 var moduleBuilder = assemblyBuilder.DefineDynamicModule("KittyModule", "Kitty.exe");

模塊即是代碼的集合,一個程序集中可以有多個模塊。並且理論上講,每個模塊可以使用不同的編程語言實現,例如C#/VB。
構造一個類型構造器,

1 // create type builder for a class
2 var typeBuilder = moduleBuilder.DefineType("HelloKittyClass", TypeAttributes.Public);

通過類型構造器定義一個方法,獲取方法構造器,獲得方法構造器的IL生成器,通過編寫IL代碼來定義方法功能。

技術分享圖片
 1 // create method builder
 2 var methodBuilder = typeBuilder.DefineMethod(
 3   "SayHelloMethod",
 4   MethodAttributes.Public | MethodAttributes.Static,
 5   null,
 6   null);
 7  
 8 // then get the method il generator
 9 var il = methodBuilder.GetILGenerator();
10  
11 // then create the method function
12 il.Emit(OpCodes.Ldstr, "Hello, Kitty!");
13 il.Emit(OpCodes.Call, 
14   typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
15 il.Emit(OpCodes.Call, typeof(Console).GetMethod("ReadLine"));
16 il.Emit(OpCodes.Pop); // we just read something here, throw it.
17 il.Emit(OpCodes.Ret);
技術分享圖片

創建類型,

1 // then create the whole class type
2 var helloKittyClassType = typeBuilder.CreateType();

如果當前程序集是可運行的,則設置一個程序入口,

1 // set entry point for this assembly
2 assemblyBuilder.SetEntryPoint(helloKittyClassType.GetMethod("SayHelloMethod"));

將動態生成的程序集保存成磁盤文件,

1 // save assembly
2 assemblyBuilder.Save("Kitty.exe");

此時,通過反編譯工具,將Kitty.exe反編譯成代碼,

技術分享圖片
 1 using System;
 2 
 3 public class HelloKittyClass
 4 {
 5   public static void SayHelloMethod()
 6   {
 7     Console.WriteLine("Hello, Kitty!");
 8     Console.ReadLine();
 9   }
10 }
技術分享圖片

運行結果,
技術分享圖片

完整代碼

技術分享圖片
 1 using System;
 2 using System.Reflection;
 3 using System.Reflection.Emit;
 4 
 5 namespace EmitIntroduction
 6 {
 7   class Program
 8   {
 9     static void Main(string[] args)
10     {
11       // specify a new assembly name
12       var assemblyName = new AssemblyName("Kitty");
13 
14       // create assembly builder
15       var assemblyBuilder = AppDomain.CurrentDomain
16         .DefineDynamicAssembly(assemblyName, 
17           AssemblyBuilderAccess.RunAndSave);
18 
19       // create module builder
20       var moduleBuilder = 
21         assemblyBuilder.DefineDynamicModule(
22           "KittyModule", "Kitty.exe");
23 
24       // create type builder for a class
25       var typeBuilder = 
26         moduleBuilder.DefineType(
27           "HelloKittyClass", TypeAttributes.Public);
28 
29       // create method builder
30       var methodBuilder = typeBuilder.DefineMethod(
31         "SayHelloMethod",
32         MethodAttributes.Public | MethodAttributes.Static,
33         null,
34         null);
35 
36       // then get the method il generator
37       var il = methodBuilder.GetILGenerator();
38 
39       // then create the method function
40       il.Emit(OpCodes.Ldstr, "Hello, Kitty!");
41       il.Emit(OpCodes.Call, 
42         typeof(Console).GetMethod(
43         "WriteLine", new Type[] { typeof(string) }));
44       il.Emit(OpCodes.Call, 
45         typeof(Console).GetMethod("ReadLine"));
46       il.Emit(OpCodes.Pop); // we just read something here, throw it.
47       il.Emit(OpCodes.Ret);
48 
49       // then create the whole class type
50       var helloKittyClassType = typeBuilder.CreateType();
51 
52       // set entry point for this assembly
53       assemblyBuilder.SetEntryPoint(
54         helloKittyClassType.GetMethod("SayHelloMethod"));
55 
56       // save assembly
57       assemblyBuilder.Save("Kitty.exe");
58 
59       Console.WriteLine(
60         "Hi, Dennis, a Kitty assembly has been generated for you.");
61       Console.ReadLine();
62     }
63   }
64 }
技術分享圖片

下載完整代碼

進一步閱讀使用Emit生成構造函數和屬性

秒懂C#通過Emit動態生成代碼