秒懂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,如需转载请自行联系原作者
目录
相关文章
|
C# Windows
C#通过代码实现快捷键编辑
C#通过代码实现快捷键编辑
219 1
|
3月前
|
机器学习/深度学习 算法 定位技术
Baumer工业相机堡盟工业相机如何通过YoloV8深度学习模型实现裂缝的检测识别(C#代码UI界面版)
本项目基于YOLOv8模型与C#界面,结合Baumer工业相机,实现裂缝的高效检测识别。支持图像、视频及摄像头输入,具备高精度与实时性,适用于桥梁、路面、隧道等多种工业场景。
323 27
|
12月前
|
缓存 C# Windows
C#程序如何编译成Native代码
【10月更文挑战第15天】在C#中,可以通过.NET Native和第三方工具(如Ngen.exe)将程序编译成Native代码,以提升性能和启动速度。.NET Native适用于UWP应用,而Ngen.exe则通过预编译托管程序集为本地机器代码来加速启动。不过,这些方法也可能增加编译时间和部署复杂度。
620 2
|
C# 开发者 Windows
在VB.NET项目中使用C#编写的代码
在VB.NET项目中使用C#编写的代码
161 0
|
12月前
|
C#
C# 图形验证码实现登录校验代码
C# 图形验证码实现登录校验代码
282 2
|
前端开发 程序员 API
从后端到前端的无缝切换:一名C#程序员如何借助Blazor技术实现全栈开发的梦想——深入解析Blazor框架下的Web应用构建之旅,附带实战代码示例与项目配置技巧揭露
【8月更文挑战第31天】本文通过详细步骤和代码示例,介绍了如何利用 Blazor 构建全栈 Web 应用。从创建新的 Blazor WebAssembly 项目开始,逐步演示了前后端分离的服务架构设计,包括 REST API 的设置及 Blazor 组件的数据展示。通过整合前后端逻辑,C# 开发者能够在统一环境中实现高效且一致的全栈开发。Blazor 的引入不仅简化了 Web 应用开发流程,还为习惯于后端开发的程序员提供了进入前端世界的桥梁。
1646 1
|
12月前
|
中间件 数据库连接 API
C#数据分表核心代码
C#数据分表核心代码
125 0
|
物联网 C# Windows
看看如何使用 C# 代码让 MQTT 进行完美通信
看看如何使用 C# 代码让 MQTT 进行完美通信
1438 0
|
数据安全/隐私保护 C# UED
利用 Xamarin 开展企业级移动应用开发:从用户登录到客户管理,全面演示C#与Xamarin.Forms构建跨平台CRM应用的实战技巧与代码示例
【8月更文挑战第31天】利用 Xamarin 进行企业级移动应用开发能显著提升效率并确保高质量和高性能。Xamarin 的跨平台特性使得开发者可以通过单一的 C# 代码库构建 iOS、Android 和 Windows 应用,帮助企业快速推出产品并保持一致的用户体验。本文通过一个简单的 CRM 示例应用演示 Xamarin 的使用方法,并提供了具体的代码示例。该应用包括用户登录、客户列表显示和添加新客户等功能。此外,还介绍了如何增强应用的安全性、数据持久化、性能优化及可扩展性,从而构建出功能全面且体验良好的移动应用。
172 0
|
前端开发 开发者 Apache
揭秘Apache Wicket项目结构:如何打造Web应用的钢铁长城,告别混乱代码!
【8月更文挑战第31天】Apache Wicket凭借其组件化设计深受Java Web开发者青睐。本文详细解析了Wicket项目结构,帮助你构建可维护的大型Web应用。通过示例展示了如何使用Maven管理依赖,并组织页面、组件及业务逻辑,确保代码清晰易懂。Wicket提供的页面继承、组件重用等功能进一步增强了项目的可维护性和扩展性。掌握这些技巧,能够显著提升开发效率,构建更稳定的Web应用。
238 0

热门文章

最新文章