C#如何用IL和Emit类通过Calli来实现实例函数与静态函数的调用
一. 介绍
最近充能看书,在书上看到函数调用可以 " 通过 ldftn 获得函数指针,然后使用 calli 指令 " 来进行调用,并说这种行为 " 类似 C 的函数指针,但是 C# 不支持这种行为 ",那么这是一种什么样的调用呢?我翻阅了一些资料,才知道 ldftn 和 calli 分别是 IL 语言中的两个指令 ,也就是说这是一种基于 IL 语言的调用。
事实上,C#确实不直接支持这种方式调用函数,但是却可以通过 Emit 类的相关方法来构造 IL 来间接的实现调用。那么,我们开始看看是怎么实现的吧。
二. 实现准备和实现原理
1. 首先我们要实现的方法调用,首先我们有如下类:
public class A
{
public void Test(int num)
{
Console.WriteLine(num);
} public static void Test2(int num)
{
Console.WriteLine(num);
}
}
2. IL语言相关:
(1)Ldftn 指令:
- 语法:ldftn <token>
- 功能:把函数指针加载到由 MethodDef 或 MemberRef 类型的 <token> 所指定的方法上
(2)Calli 指令:
- 语法:calli <token>
- 功能:先从栈上弹出函数指针,再从栈上弹出所有的参数,然后根据 <toke> 指定的方法签名进行间接方法调用。<token> 必须是有效的 StandAloneSig 标记。函数指针必须位于栈顶。如过方法返回数值,那么该数值在调用完成后被压入栈上。
★ 3. IL调用函数方法的一般流程:
一般在IL中使用方式分为两大类,一种是调用托管函数,另外一种是调用非托管的函数,这里我们暂不考虑对非托管函数的调用。对于托管函数,按照其调用方法是不是要引用一个对象实列,可以分为静态方法和实例方法。
接下来就对这两种类型的方法,分别描述一下 IL 代码的一般流程。
(1)实例方法:
过程如下:
- 通过 " newobj instance 构造函数签名 " 先创建一个实例对象,并将对象从栈顶弹出保存到局部变量
- 通过 " ldftn instance 实例方法签名 " 来获得函数方法的指针,调用完成后这个指针会被放置于栈顶,同理也将栈顶的函数指针弹出并保存到局部变量
- 把实例对象放置到栈顶,由于实例方法要 this 指向的对象,所以这个对象会作为第一个参数 arg.0 作为 this
- 按照函数的调用的参数的顺序,依次将变量放置到栈顶
- 把函数的指针放置到栈顶
- 通过 " calli instance 实例方法签名 " 来进行函数的调用,调用完成之后会把返回值放置与栈顶,最后把栈顶的返回值弹出并保存使用
(2)静态方法:
静态方法不需要 this 指向的对象,所以这边也不需要先创建一个对象,过程如下:
- 通过 " ldftn 静态方法签名 " 来获得函数方法的指针,调用完成后这个指针会被放置于栈顶,将栈顶的函数指针弹出并保存到局部变量
- 按照函数的调用的参数的顺序,依次将变量放置到栈顶
- 把函数的指针放置到栈顶
- 通过 " calli 静态方法签名 " 来进行函数的调用,调用完成之后会把返回值放置与栈顶,最后把栈顶的返回值弹出并保存使用
(3)一个简单的实例调用的例子:
.locals init (native int fnptr)
...
ldfrn void [mscorlib]System.Console::WriteLine(int32)
stloc. //本地变量中存储函数指针
...
ldc.i4 //加载参数
ldloc.
callo void(int32)
...
下面我们看一下怎么实现对 Test 和 Test2 的调用的,直接上菜...
三. IL 代码实现
IL 代码如下:
.assembly extern mscorlib
{
auto
}
.assembly MyTest {}
.module MyTest.exe
.class public A
{
.method public specialname void .ctor()
{
ldarg.
call instance void [mscorlib]System.Object::.ctor()
ret
} .method public void Test(int32 param_0)
{
ldarg.
call void [mscorlib]System.Console::WriteLine(int32)
ret
} .method public static void Test2(int32 param_0)
{
ldarg.
call void [mscorlib]System.Console::WriteLine(int32)
ret
}
}
.method public static void Main()
{
.entrypoint
.locals (class A a,int32 v_1)
//实例方法调用
newobj instance void A::.ctor()
stloc.
ldftn instance void A::Test(int32)
stloc.
ldloc.
ldc.i4
ldloc.
calli instance void(int32) //静态方法调用
ldftn void A::Test2(int32)
stloc.
ldc.i4
ldloc.
calli void(int32)
ret
}
四. C# 用 Emit 类来实现
C# 代码:
public class CreateTypeHelper
{
public static Type CreateMethodCallingType()
{
//获得当前的程序域
AppDomain currentDomain = Thread.GetDomain();
//创建这个类的程序集
AssemblyName assemblyName = new AssemblyName();
assemblyName.Name = "DynamicAssembly";
AssemblyBuilder assemblyBuilder = currentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
//创建模块
ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule("MethodCallingModule", "MethodCalling.dll");
//创建类型
TypeBuilder typeBuilder = moduleBuilder.DefineType("MethodCalling", TypeAttributes.Public);
//创建一个方法
MethodBuilder methodBuilder = typeBuilder.DefineMethod("CalliMethodCall", MethodAttributes.Public | MethodAttributes.Static, typeof(void), new Type[]);
//IL
ILGenerator il = methodBuilder.GetILGenerator();
//.locals (class A a,int32 v_1)
il.DeclareLocal(typeof(A));
il.DeclareLocal(typeof(Int32)); //newobj instance void A::.ctor()
ConstructorInfo constructorInfo = typeof(A).GetConstructor(new Type[]);
il.Emit(OpCodes.Newobj, constructorInfo);
//stloc.0
il.Emit(OpCodes.Stloc_0);
//获得A.Test方法
MethodInfo methodInfo = typeof(A).GetMethod("Test", BindingFlags.Public | BindingFlags.Instance);
//ldftn instance void A::Test(int32)
il.Emit(OpCodes.Ldftn, methodInfo);
//stloc.1
il.Emit(OpCodes.Stloc_1);
//ldloc.0
il.Emit(OpCodes.Ldloc_0);
//ldc.i4 120
il.Emit(OpCodes.Ldc_I4, );
////ldloc.1
il.Emit(OpCodes.Ldloc_1);
//calli instance void(int32)
il.EmitCalli(OpCodes.Calli, CallingConventions.HasThis, typeof(void), new Type[] { typeof(Int32) }, null);
////获得A.Test方法
MethodInfo methodInfo1 = typeof(A).GetMethod("Test2", BindingFlags.Public | BindingFlags.Static);
//ldftn void A::Test2(int32)
il.Emit(OpCodes.Ldftn, methodInfo1);
//stloc.1
il.Emit(OpCodes.Stloc_1);
//ldc.i4 233
il.Emit(OpCodes.Ldc_I4, );
//ldloc.1
il.Emit(OpCodes.Ldloc_1);
//calli void(int32)
il.EmitCalli(OpCodes.Calli,CallingConventions.Standard, typeof(void), new Type[] { typeof(Int32) }, null);
//ret
il.Emit(OpCodes.Ret); Type retType = typeBuilder.CreateType();
assemblyBuilder.Save("MethodCalling.dll"); return retType;
}
}
上端调用方法:
Type type = CreateTypeHelper.CreateMethodCallingType();
//获得方法
MethodInfo methodInfo = type.GetMethod("CalliMethodCall");
if (methodInfo != null)
{
methodInfo.Invoke(null, null);
}
执行结果:
五. 验证结果
最后我们验证一下动态生成的类。在代码中,我们创建了一个 " MethodCalling.dll " 用来保存动态生成的类,里面承载了我们的 Emit 代码生成的 IL代码,如下图:
用 ILDasm.exe 查看后如下图:
看了一下与我们写的 IL 代码基本一致,今天对 Calli 调用函数方法的研究基本成功!
C#如何用IL和Emit类通过Calli来实现实例函数与静态函数的调用的更多相关文章
- 如何用ActiveQt写导出类
如何用ActiveQt写导出类 最近一直在用ActiveQt框架来写ActiveX插件, 由于项目需要提示类的导出, 所以上午捣鼓了一下, 现在记录记录.其实内容主要是把Qt手册里自己用到的部分整理一 ...
- 【转载】 C++多继承中重写不同基类中相同原型的虚函数
本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数. 在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数.下面就是一个例子: class CBaseA { public: ...
- 类中用const限定的成员函数
本文转自http://blog.csdn.net/whyglinux/article/details/602329 类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非 ...
- 使用iskindofclass来发现对象是否是某类或其子类的实例
发现对象是否是特定类或其子类的实例 要发现对象是否是某类或其子类的实例,请在对象上调用 isKindOfClass: 方法.当应用程序需要发现其响应的消息(实现的或继承的),它有时进行以上的检查. s ...
- C++@类的静态成员变量和静态成员函数
参考: http://blog.csdn.net/morewindows/article/details/6721430 http://www.cnblogs.com/lzjsky/archive/2 ...
- 基类中定义的虚函数在派生类中重新定义时,其函数原型,包括返回类型、函数名、参数个数、参数类型及参数的先后顺序,都必须与基类中的原型完全相同 but------> 可以返回派生类对象的引用或指针
您查询的关键词是:c++primer习题15.25 以下是该网页在北京时间 2016年07月15日 02:57:08 的快照: 如果打开速度慢,可以尝试快速版:如果想更新或删除快照,可以投诉快照. ...
- 用C++设计一个不能被继承的类(用私有构造函数+友元函数)
题目:用C++设计一个不能被继承的类. 分析:这是Adobe公司2007年校园招聘的最新笔试题.这道题除了考察应聘者的C++基本功底外,还能考察反应能力,是一道很好的题目. 在Java中定义了关键字f ...
- Python笔记(七):字典、类、属性、对象实例、继承
(一) 简单说明 字典是Python的内置数据结构,将数据与键关联(例如:姓名:张三,姓名是键,张三就是数据).例如:下面这个就是一个字典 {'姓名': '张三', '出生日期': '2899-08 ...
- python基础——类名称空间与对象(实例)名称空间
python基础--类名称空间与对象(实例)名称空间 1 类名称空间 创建一个类就会创建一个类的名称空间,用来存储类中定义的所有名字,这些名字称为类的属性 而类的良好总属性:数据属性和函数属性 其中类 ...
随机推荐
- Deepnude算法“tuo”衣服
PS:我不是偷窥狂.我是技术的爱好者 换脸视频后AI又出偏门应用:用算法“tuo”女性衣服 据美国科技媒体Motherboard报道,一名程序员最近开发出一款名叫DeepNude的应用,只要给Deep ...
- 【NOI 2011】阿狸的打字机
Problem Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有 \(28\) 个按键,分别印有 \(26\) 个小写英文字母和 B . P 两个字母. ...
- Windows下第一个驱动程序
Windows内核分析索引目录:https://www.cnblogs.com/onetrainee/p/11675224.html 参考下面博客: VS2017搭建驱动开发环境WDK :https: ...
- Spring注解式AOP面向切面编程.
1.AOP指在程序运行期间动态的将某段代码切入到指定方法指定位置进行运行的编程方式.aop底层是动态代理. package com.bie.config; import org.aspectj.lan ...
- JavaScript学习笔记-----NaN、isNan
NaN / Number.NaN 全局属性 NaN 的值表示不是一个数字(Not-A-Number), NaN 属性的初始值就是 NaN,和 Number.NaN 的值一样. 在现代浏览器中(ES ...
- Linux下使用 github+hexo 搭建个人博客01-hexo搭建
为什么要搭建自己的博客系统? 原因有好几个吧,归类如下:1.自己搭建博客系统很有成就感,可以自己选定页面风格和页面排版: 2.自己搭建博客系统可以根据自己的需要添加各种插件功能,因此整体上比网上的第三 ...
- Linux:LAMP环境的搭建
LAMP环境的搭建 安装DNS服务器 安装DNS服务 yum install bind -y DNS的配置 创建正向解析 以创建一个名为"lsy.com"的正向查找区域为例: 第一 ...
- 初级模拟电路:3-2 BJT的工作原理
回到目录 和前面介绍二极管的PN结的工作原理一样,BJT的量子级工作机制也非常复杂,一般教科书上为了帮助学习者能快速理解,也都是用一种简化模型的方法来介绍BJT的工作机理,一般只需大致了解即可.只要记 ...
- (办公)访问其他系统接口httpClient,异步访问
访问其他系统接口httpClient,但是都是同步的,同步意味当前线程是阻塞的,只有本次请求完成后才能进行下一次请求;异步意味着所有的请求可以同时塞入缓冲区,不阻塞当前的线程; httpClient请 ...
- 安装pymssql
直接安装失败 https://www.lfd.uci.edu/~gohlke/pythonlibs/#pymssql 去下载对应的 pymssql whl版本 之后 pip install whe ...