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 框架为服务接口生成动态代理类,接着再去调用代理类实现远程接口调用.在阅读这部分源码时,最后 ...
随机推荐
- configparser模块--配置文件
该模块适用于配置文件的格式与windows ini文件类似,可以包含一个或多个节(section),每个节可以有多个参数(键=值). 创建文件 import configparser config = ...
- ECC加密算法入门介绍 --- 看雪
标 题:ECC加密算法入门介绍 作 者:zmworm 时 间:2003/05/04 08:32pm 链 接:http://bbs.pediy.com ECC加密算法入门介绍 作者 :ZMWorm[C ...
- C#红绿状态灯
1.在Label里 画圆,存在窗体刷新会丢失画. public void SetShowConnectStatus(Label lbl, bool isOk) { lbl.Text = "& ...
- ubuntu server 14.04 上安装jdk1.8
ubuntu server 14.04 上安装jdk1.8 1.使用apt-get安装oracle-jdk安装oracle jdk sudo apt-get install python-softwa ...
- aview安装和使用
一.安装aalibwget https://sourceforge.net/projects/aa-project/files/latest/download?source=files --no-ch ...
- 缺失dll的问题
不小心运行一下什么程序就会出现缺失xxx.dll的问题,太烦了,遇到好多,一直没有记录.现在开始记录,以便日后查看~ 1. api-ms-win-crt-runtime-l1-1-0.dll 64位系 ...
- PID控制器开发笔记之十一:专家PID控制器的实现
前面我们讨论了经典的数字PID控制算法及其常见的改进与补偿算法,基本已经覆盖了无模型和简单模型PID控制经典算法的大部.再接下来的我们将讨论智能PID控制,智能PID控制不同于常规意义下的智能控制,是 ...
- 用C#开发基于自动化接口的OPC客户端
OPC全称是Object Linking and Embedding(OLE) for Process Control,它的出现为基于Windows的应用程序和现场过程控制应用建立了桥梁.OPC作为一 ...
- windows 系统纯净版官网下载地址
http://www.imsdn.cn/operating-systems/windows-7/
- JavaScript从入门到精通(附光盘1张):作者:明日科技出版社:清华大学出版社出版时间:2012年09月
本书介绍 一:本书 pdf 获取信息 本书下载:请申请加入本群 (QQ群:668345923), 并联系群主. 本群主有:本书pdf 全文教材 及附带的 光盘内容 二:本书目录介绍 第1篇 基 ...