C#使用Emit构造拦截器动态代理类
在AOP编程概念介绍中,常见的示例为拦截对象,并在对象的某方法执行前和执行后分别记录日志。
而最常用的拦截方式是使用动态代理类,用其封装一个日志拦截器,当方法被执行时进行日志记录。
日志拦截器类

1 public class Interceptor
2 {
3 public object Invoke(object @object, string @method, object[] parameters)
4 {
5 Console.WriteLine(
6 string.Format("Interceptor does something before invoke [{0}]...", @method));
7
8 var retObj = @object.GetType().GetMethod(@method).Invoke(@object, parameters);
9
10 Console.WriteLine(
11 string.Format("Interceptor does something after invoke [{0}]...", @method));
12
13 return retObj;
14 }
15 }

被拦截对象类
假设我们有一个Command类,包含一个方法Execute用于执行一些工作。

1 public class Command
2 {
3 public virtual void Execute()
4 {
5 Console.WriteLine("Command executing...");
6 Console.WriteLine("Hello Kitty!");
7 Console.WriteLine("Command executed.");
8 }
9 }

我们需要在Execute方法执行前和执行后分别记录日志。
动态代理类

1 public class Proxy
2 {
3 public static T Of<T>() where T : class, new()
4 {
5 string nameOfAssembly = typeof(T).Name + "ProxyAssembly";
6 string nameOfModule = typeof(T).Name + "ProxyModule";
7 string nameOfType = typeof(T).Name + "Proxy";
8
9 var assemblyName = new AssemblyName(nameOfAssembly);
10 var assembly = AppDomain.CurrentDomain
11 .DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
12 var moduleBuilder = assembly.DefineDynamicModule(nameOfModule);
13
14 var typeBuilder = moduleBuilder.DefineType(
15 nameOfType, TypeAttributes.Public, typeof(T));
16
17 InjectInterceptor<T>(typeBuilder);
18
19 var t = typeBuilder.CreateType();
20
21 return Activator.CreateInstance(t) as T;
22 }
23
24 private static void InjectInterceptor<T>(TypeBuilder typeBuilder)
25 {
26 // ---- define fields ----
27
28 var fieldInterceptor = typeBuilder.DefineField(
29 "_interceptor", typeof(Interceptor), FieldAttributes.Private);
30
31 // ---- define costructors ----
32
33 var constructorBuilder = typeBuilder.DefineConstructor(
34 MethodAttributes.Public, CallingConventions.Standard, null);
35 var ilOfCtor = constructorBuilder.GetILGenerator();
36
37 ilOfCtor.Emit(OpCodes.Ldarg_0);
38 ilOfCtor.Emit(OpCodes.Newobj, typeof(Interceptor).GetConstructor(new Type[0]));
39 ilOfCtor.Emit(OpCodes.Stfld, fieldInterceptor);
40 ilOfCtor.Emit(OpCodes.Ret);
41
42 // ---- define methods ----
43
44 var methodsOfType = typeof(T).GetMethods(BindingFlags.Public | BindingFlags.Instance);
45
46 for (var i = 0; i < methodsOfType.Length; i++)
47 {
48 var method = methodsOfType[i];
49 var methodParameterTypes =
50 method.GetParameters().Select(p => p.ParameterType).ToArray();
51
52 var methodBuilder = typeBuilder.DefineMethod(
53 method.Name,
54 MethodAttributes.Public | MethodAttributes.Virtual,
55 CallingConventions.Standard,
56 method.ReturnType,
57 methodParameterTypes);
58
59 var ilOfMethod = methodBuilder.GetILGenerator();
60 ilOfMethod.Emit(OpCodes.Ldarg_0);
61 ilOfMethod.Emit(OpCodes.Ldfld, fieldInterceptor);
62
63 // create instance of T
64 ilOfMethod.Emit(OpCodes.Newobj, typeof(T).GetConstructor(new Type[0]));
65 ilOfMethod.Emit(OpCodes.Ldstr, method.Name);
66
67 // build the method parameters
68 if (methodParameterTypes == null)
69 {
70 ilOfMethod.Emit(OpCodes.Ldnull);
71 }
72 else
73 {
74 var parameters = ilOfMethod.DeclareLocal(typeof(object[]));
75 ilOfMethod.Emit(OpCodes.Ldc_I4, methodParameterTypes.Length);
76 ilOfMethod.Emit(OpCodes.Newarr, typeof(object));
77 ilOfMethod.Emit(OpCodes.Stloc, parameters);
78
79 for (var j = 0; j < methodParameterTypes.Length; j++)
80 {
81 ilOfMethod.Emit(OpCodes.Ldloc, parameters);
82 ilOfMethod.Emit(OpCodes.Ldc_I4, j);
83 ilOfMethod.Emit(OpCodes.Ldarg, j + 1);
84 ilOfMethod.Emit(OpCodes.Stelem_Ref);
85 }
86 ilOfMethod.Emit(OpCodes.Ldloc, parameters);
87 }
88
89 // call Invoke() method of Interceptor
90 ilOfMethod.Emit(OpCodes.Callvirt, typeof(Interceptor).GetMethod("Invoke"));
91
92 // pop the stack if return void
93 if (method.ReturnType == typeof(void))
94 {
95 ilOfMethod.Emit(OpCodes.Pop);
96 }
97
98 // complete
99 ilOfMethod.Emit(OpCodes.Ret);
100 }
101 }
102 }

使用动态代理类

1 class Program
2 {
3 static void Main(string[] args)
4 {
5 var command = Proxy.Of<Command>();
6 command.Execute();
7
8 Console.WriteLine("Hi, Dennis, great, we got the interceptor works.");
9 Console.ReadLine();
10 }
11 }

运行结果

完整代码
C#使用Emit构造拦截器动态代理类的更多相关文章
- 秒懂C#通过Emit动态生成代码 C#使用Emit构造拦截器动态代理类
秒懂C#通过Emit动态生成代码 首先需要声明一个程序集名称, 1 // specify a new assembly name 2 var assemblyName = new Assembly ...
- 【Java EE 学习 75 下】【数据采集系统第七天】【二进制运算实现权限管理】【使用反射初始化权限表】【权限捕获拦截器动态添加权限】
一.使用反射动态添加权限 在该系统中,我使用struts2的时候非常规范,访问的Action的形式都是"ActionClassName_MethodName.action?参数列表" ...
- .Net基础——程序集与CIL HttpClient封装方法 .Net Core 编码规范 C#中invoke和beginInvoke的使用 WebServeice 动态代理类
.Net基础——程序集与CIL 1. 程序集和CIL: 程序集是由.NET语言的编译器接受源代码文件产生的输出文件,通常分为 exe和dll两类,其中exe包含Main入口方法可以双击执行,dll ...
- spring AbstractBeanDefinition创建bean类型是动态代理类的方式
1.接口 Class<?> resourceClass 2.获取builder BeanDefinitionBuilder builder = BeanDefinitionBuilder. ...
- WebServeice 动态代理类
1, webservice是什么? 是一个平台独立的,低耦合的,自包含的.基于可编程的web的应用程序,可使用开放的XML(标准通用标记语言下的一个子集)标准来描述.发布.发现.协调和配置这些应用程序 ...
- CGLIB 和 JDK生成动态代理类的区别(转)
文章转自http://luyuanliang.iteye.com/blog/1137292 AOP 使用的设计模式就是代理模式,是对IOC设计的补充.为了扩展性,往往会加上反射,动态生成字节码,生成代 ...
- java代理模式及动态代理类
1. 代理模式 代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用 ...
- 初看Mybatis 源码 (二) Java动态代理类
先抛出一个问题,用过Mybatis的都知道,我们只需要定义一个Dao的接口,在里面写上一些CRUD相关操作,然后配置一下sql映射文件,就可以达到调用接口中的方法,然后执行sql语句的效果,为什么呢? ...
- Alibaba Java诊断工具Arthas查看Dubbo动态代理类
原创/朱季谦 阅读Dubbo源码过程中,会发现,Dubbo消费端在做远程调用时,默认通过 Javassist 框架为服务接口生成动态代理类,接着再去调用代理类实现远程接口调用.在阅读这部分源码时,最后 ...
随机推荐
- git与eclipse集成之代码冲突与解决
1.1. 代码冲突与解决 目前使用git管理代码,产生冲突的原因,主要是当多个人向特性分支提交代码时,如果两个人修改了同一个文件,第二个人提交代码时就可能会冲突. 举例说明: 创建远程特性分支.远程个 ...
- javascript日期格式yyyyMMddHHmmss
1. function GetDateTimeToString() { var date_ = new Date(); var year = date_.getFullYear(); ; var da ...
- zabbix在运维方面的监控方法小结
一些经典的运维问题: .配置文件中有空格,导致服务端下发的域名出现问题 .修改数据库没有备份 .修改dnspod问题,指向了错误的IP地址 .时间不一致,需要重新设定时区 .启动程序必须是最新版本,如 ...
- MySQL数据库的一些方法使用
substring_index(windSpeed,)/3.6 as windSpeed 可将 .8公里.0m/s 进行拆分 嵌套使用replace方法 replace( replace( repla ...
- iOS post提交数据有嵌套数组的处理方法
2017年11月21日17:11:43 解决办法, 修改iOS框架里的代码: http://www.jianshu.com/p/130daa0c2fe7 确实有效, 要不然, 内层的每一个键值对都会 ...
- YUV的数据格式
一.YUV格式分为两大类:planar(平面)和packed(打包).planar格式,先连续存储所有像素点的Y分量,紧接着存储所有像素点的U,随后存储所有像素点的V.packed格式,每个像素点的Y ...
- var_export 掉咋天
var_export 文件缓存经常使用 输出或返回一个变量的字符串表示 /** * 写入缓存 * * @param string $id * @param mixed $data * @ ...
- 构造函数中base与this的区别
base是对父类的引用,而this是对类本身的引用. namespace ConsoleApplication1 { public class BaseClass { private string n ...
- Ex 2_23 如果一个数组超过半数的元素都相同时,该数组被称为含有一个主元素..._第二次作业
将数组A划分为两个数组A1和A2 ,各含有A的一半元素或一半多一个.若A中含有主元素x,则A1和A2中至少有一个数组含有主元素x,对A1和A2递归地计算有无主元素,若A只含有一个元素,则A的主元素就是 ...
- 30)django-ORM(元信息,级联删除,正反向操作,连表查询优化)
一:元信息 class User(models.Model): name=models.CharField(max_length,index=True) email=model.CharField(m ...