秒懂C#通过Emit动态生成代码

简介:

首先需要声明一个程序集名称,

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生成构造函数和属性






本文转自匠心十年博客园博客,原文链接:http://www.cnblogs.com/gaochundong/archive/2013/06/01/csharp_emit_generate_assembly.html,如需转载请自行联系原作者
目录
相关文章
|
1月前
|
C# Windows
C#通过代码实现快捷键编辑
C#通过代码实现快捷键编辑
|
3月前
|
开发框架 .NET 编译器
C# 10.0中Lambda表达式的改进:更简洁、更灵活的代码编写体验
【1月更文挑战第21天】随着C#语言的不断发展,Lambda表达式作为一种简洁、高效的函数式编程工具,在C# 10.0中迎来了重要的改进。本文将详细探讨C# 10.0中Lambda表达式的新特性,包括参数类型的推断增强、自然类型的Lambda参数以及Lambda表达式的属性改进等。这些改进不仅简化了Lambda表达式的编写过程,还提升了代码的可读性和灵活性,为开发者带来了更优质的编程体验。
|
3月前
|
C# 开发者
C# 10.0中的文件范围命名空间:简化代码组织的新方式
【1月更文挑战第18天】C# 10.0引入了文件范围的命名空间,这是一种新的语法糖,用于更简洁地组织和管理代码。文件范围命名空间允许开发者在每个文件的基础上定义命名空间,而无需显式使用花括号包裹整个文件内容。本文将深入探讨文件范围命名空间的工作原理、使用场景以及它们为C#开发者带来的便利。
|
4月前
|
存储 人工智能 C#
【Unity 3D】C#中数组、集合、栈、队列、哈希表、字典的讲解(附测试代码)
【Unity 3D】C#中数组、集合、栈、队列、哈希表、字典的讲解(附测试代码)
36 0
|
5月前
|
开发框架 .NET C#
如何调试 C# Emit 生成的动态代码?
如何调试 C# Emit 生成的动态代码?
|
4月前
|
IDE C# 开发工具
C# | 多线程批量下载文件(创建N个线程同时批量下载文件,只需要几行代码而已)
批量下载文件时使用多线程可以有效缩短完成时间,本文将讲解如何使用C#+CodePlus扩展库快速完成多线程的文件下载。 大部分代码由IDE自动生成,需要我们自己编写的代码正好**10行**。也就是说,只需要10分钟,就可以手撸一个多线程的批量下载器。
88 0
C# | 多线程批量下载文件(创建N个线程同时批量下载文件,只需要几行代码而已)
|
2月前
|
数据采集 JSON 前端开发
从代码到内容:使用C#和Fizzler探索Instagram的深处
Instagram是一个流行的社交媒体平台,拥有数亿的用户和海量的图片和视频内容。如果您想要从Instagram上获取一些有用的信息或数据,您可能需要使用爬虫技术来自动化地抓取和分析网页内容。本文将介绍如何使用C#和Fizzler这两个强大的工具,来实现一个简单而高效的Instagram爬虫,从代码到内容,探索Instagram的深处。
|
3月前
|
存储 传感器 监控
工业相机如何实现实时和本地Raw格式图像和Bitmap格式图像的保存和相互转换(C#代码,UI界面版)
工业相机如何实现实时和本地Raw格式图像和Bitmap格式图像的保存和相互转换(C#代码,UI界面版)
31 0
|
3月前
|
存储 C# 容器
掌握 C# 变量:在代码中声明、初始化和使用不同类型的综合指南
变量是用于存储数据值的容器。 在 C# 中,有不同类型的变量(用不同的关键字定义),例如: int - 存储整数(没有小数点的整数),如 123 或 -123 double - 存储浮点数,有小数点,如 19.99 或 -19.99 char - 存储单个字符,如 'a' 或 'B'。Char 值用单引号括起来 string - 存储文本,如 "Hello World"。String 值用双引号括起来 bool - 存储具有两个状态的值:true 或 false
37 2
|
8月前
|
存储 C# 图形学
代码解析 C# 引用类型还是值类型
代码解析 C# 引用类型还是值类型