服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - 详解c# Emit技术

详解c# Emit技术

2022-10-25 12:54Min.Xiaoshuang C#

这篇文章主要介绍了c# Emit技术的相关资料,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下

我们常常有一个应用场景,由我们的C#代码,动态生成一个EXE,其应用场景可以非常多,比如软件授权,可以输入授权信息后,生成一个授权的DLL等,那如何实现这个功能呢,就要提到一个技术Emit。

1、Emit概述

Emit,可以称为发出或者产生。在Framework中,与Emit相关的类基本都存在于System.Reflection.Emit命名
空间下。可见Emit是作为反射的一个元素存在的。说道反射,大家应该都不陌生,它允许我们查看程序集的元素据,从而取得形如程序集包含哪些类型,类型包
含哪些方法等等大量的信息。但是反射也仅能够‘看',而Emit则可以在运行时动态生成代码。接下来就来看看如何用Emit生成代码。

2、程序集(Assembly)和模块(Managed Module)

程序集是一个或多个模块、资源文件的逻辑性分组,其次程序集是重用,安全性和版本控制的最小单元。我们所见到的DLL、EXE都可以称为一个Assembly,一个Assembly里面包含多个Module,不过通常,我们VS编译的时候,会只编译一个Module,假如在一个Assembly中要编译多个Module,则要借助csc.exe实现。

3、动态生成代码操作

定义程序集

?
1
2
3
4
5
//定义一个程序集的名称
 var asmName = new AssemblyName("MyClass");
 
//首先就需要定义一个程序集
 var defAssembly = AppDomain.CurrentDomain.DefineDynamicAssembly(asmName, AssemblyBuilderAccess.RunAndSave);

定义模块,和指定程序集的保存名称

?
1
2
//定义一个构建类
var defModuleBuilder = defAssembly.DefineDynamicModule("MyModule", "MyAssembly.dll");

定义一个类 和方法

?
1
2
3
4
5
6
7
8
//定义一个类
 var defClassBuilder =defModuleBuilder.DefineType("MyClass", TypeAttributes.Public);
 //定义一个方法
var methodBldr = defClassBuilder.DefineMethod("MyMethod",
  MethodAttributes.Public,
   null,//返回类型
   null//参数的类型
 );

以上通过创建,已经确定了程序集和模块,也定义了当前模块中的一个类和方法,但这个类的MyMethod方法只定义了一个声明,并没有定义实体操作,以下就需要应用到Emit技术中一个技术OpCode。

OpCode 是描述中间语言 (IL) 指令。这个指令非常多,可以查看微软官网:https://docs.microsoft.com/zh-cn/dotnet/api/system.reflection.emit.opcodes?view=netframework-4.8

通过OpCode我们可以定义方法的内容如下:

?
1
2
3
4
5
6
7
8
//获取IL生成器
      var il = defMethodBuilder.GetILGenerator();
      //定义一个字符串
      il.Emit(OpCodes.Ldstr, "生成的第一个程序");
      //调用一个函数
      il.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
      //返回到方法开始(返回)
      il.Emit(OpCodes.Ret);

通过以上的定义,我们完成了一个程序集、模块、类和方法的定义,我们怎么把以上的定义的信息进行创建和保存,需要调用以下函数:

?
1
2
3
4
5
//创建类型
  defClassBuilder.CreateType();
 
  //保存程序集
  defAssembly.Save("MyAssemblydll");

我们可以在运行程序看到如下效果:

详解c# Emit技术

以下通过创建程序集,并且调用的代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Reflection.Emit;
using System.Text;
using System.Threading.Tasks;
 
namespace ConsoleApp1
{
  class Program
  {
    static void Main(string[] args)
    {
 
      CreateAssembly();
      LoadAssembly();
 
      Console.ReadKey();
    }
 
    public static void LoadAssembly()
    {
      var ass = AppDomain.CurrentDomain.Load("MyAssembly");
      var m = ass.GetModule("MyModule");
      var ts = m.GetTypes();
      var t = ts.FirstOrDefault();
      if (t != null)
      {
        object obj = Activator.CreateInstance(t);
        var me = t.GetMethod("MyMethod");
        me.Invoke(obj, null);
      }
    }
    public static void CreateAssembly()
    {
 
 
      //定义一个程序集的名称
      var asmName = new AssemblyName("MyAssembly");
 
      //首先就需要定义一个程序集
      var defAssembly = AppDomain.CurrentDomain.DefineDynamicAssembly(asmName, AssemblyBuilderAccess.RunAndSave);
      //定义一个构建类
      var defModuleBuilder = defAssembly.DefineDynamicModule("MyModule", "MyAssembly.dll");
 
 
      //定义一个类
      var defClassBuilder = defModuleBuilder.DefineType("MyClass", TypeAttributes.Public);
 
      //定义一个方法
      var defMethodBuilder = defClassBuilder.DefineMethod("MyMethod",
        MethodAttributes.Public,
        null,//返回类型
        null//参数类型
        );
      //获取IL生成器
      var il = defMethodBuilder.GetILGenerator();
      //定义一个字符串
      il.Emit(OpCodes.Ldstr, "生成的第一个程序");
      //调用一个函数
      il.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
      //返回到方法开始(返回)
      il.Emit(OpCodes.Ret);
 
      //创建类型
      defClassBuilder.CreateType();
 
      //保存程序集
      defAssembly.Save("MyAssembly.dll");
    }
 
  }
}

显示效果如下:

详解c# Emit技术

以上就是详解c# Emit技术的详细内容,更多关于c# Emit技术的资料请关注服务器之家其它相关文章!

原文链接:https://www.cnblogs.com/minhost/p/12190865.html

延伸 · 阅读

精彩推荐
  • C#c# 播放声音的四种方法

    c# 播放声音的四种方法

    这篇文章主要介绍了c# 播放声音的四种方法,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下...

    搜源网10412022-10-18
  • C#C#利用时间和随即字符串创建唯一的订单编号

    C#利用时间和随即字符串创建唯一的订单编号

    本文介绍了利用时间和随机字符串组合生成唯一订单号的示例,从而保证订单号不会重复,希望能够对大家有所帮助。...

    Darren Ji4062021-11-17
  • C#unity实现虚拟摇杆控制Virtual Joystick

    unity实现虚拟摇杆控制Virtual Joystick

    这篇文章主要为大家详细介绍了unity实现虚拟摇杆控制Virtual Joystick,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    代码黑洞_9882022-09-01
  • C#实例详解C#实现http不同方法的请求

    实例详解C#实现http不同方法的请求

    本篇文章给大家分享了C#实现http不同方法的请求的相关知识点以及实例代码,有需要的朋友参考下。...

    郝光明10722022-02-25
  • C#C#通过重写Panel改变边框颜色与宽度的方法

    C#通过重写Panel改变边框颜色与宽度的方法

    这篇文章主要介绍了C#通过重写Panel改变边框颜色与宽度的方法,涉及C#针对Panel控件的重写与属性设置技巧,具有一定参考借鉴价值,需要的朋友可以参考下...

    我心依旧4442021-10-20
  • C#c# BackgroundWorker使用方法

    c# BackgroundWorker使用方法

    这篇文章主要介绍了c# BackgroundWorker使用方法,文中代码非常详细,帮助大家更好的参考学习,感兴趣的朋友可以了解下...

    风情单车5922022-09-09
  • C#详解C# 反射(Reflection)

    详解C# 反射(Reflection)

    这篇文章主要介绍了C# 反射(Reflection)的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下...

    菜鸟教程3952022-09-21
  • C#C#装箱和拆箱操作实例分析

    C#装箱和拆箱操作实例分析

    这篇文章主要介绍了C#装箱和拆箱操作,结合实例形式分析了C#中装箱与拆箱的概念、用法及相关注意事项,需要的朋友可以参考下...

    Quber6682021-12-02