博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
.NET:动态代理的 “5 + 1” 模式
阅读量:7098 次
发布时间:2019-06-28

本文共 6830 字,大约阅读时间需要 22 分钟。

背景

什么叫“动态代理”,代理模式我们都知道,动态代理就是动态生成的代理(采用Emit)。

重量级的ORM和IOC产品离不开动态代理,作为开发人员,多数情况不用关注动态代理的内部实现机制,但是了解其一般的规律和模式还是有必要的,比如:虽然你开发期间采用了POCO,因为开启了动态代理,运行期间则不是POCO。

本文简单描述了5种代理生成模式和1种Mixin模式,最后给出一个示例。

公共代码

这里先给出公共代码。

1     public interface IPlayable 2     { 3         void Play(); 4     } 5  6     public class Animal : IPlayable 7     { 8         public virtual void Play() 9         {10             Console.WriteLine("Animal.Play");11         }12     }13 14     public class Dog : Animal15     {16         public override void Play()17         {18             Console.WriteLine("Dog.Play");19         }20     }21 22     public interface IRunable23     {24         void Run();25     }26 27     public class RunAbility : IRunable28     {29         public void Run()30         {31             Console.WriteLine("RunAbility.Run");32         }33     }34 35     public class AnimalInterceptor : IInterceptor36     {37         public void Intercept(IInvocation invocation)38         {39             Console.WriteLine("Before AnimalInterceptor.Intercept");40             if (invocation.InvocationTarget != null)41             {42                 invocation.Proceed();43             }44             Console.WriteLine("After AnimalInterceptor.Intercept");45         }46     }

5种代理模式

第一种:ClassProxy

代码示例

1             { 2                 Console.WriteLine("\n*************ClassProxy*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var animal = generator.CreateClassProxy
(new AnimalInterceptor()); 5 animal.Play(); 6 7 Console.WriteLine(animal.GetType()); 8 Console.WriteLine(animal.GetType().BaseType); 9 10 var compositeField = animal.GetType().GetField("__target");11 Console.WriteLine(compositeField);12 13 foreach (var interfaceType in animal.GetType().GetInterfaces())14 {15 Console.WriteLine(interfaceType);16 }17 }

运行结果

动态代理类图

等待上传中。

第二种:ClassProxyWithTarget

代码示例

1             { 2                 Console.WriteLine("\n*************ClassProxyWithTarget*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var animal = generator.CreateClassProxyWithTarget
(new Dog(), new AnimalInterceptor()); 5 animal.Play(); 6 7 Console.WriteLine(animal.GetType()); 8 Console.WriteLine(animal.GetType().BaseType); 9 10 var compositeField = animal.GetType().GetField("__target");11 Console.WriteLine(compositeField);12 13 foreach (var interfaceType in animal.GetType().GetInterfaces())14 {15 Console.WriteLine(interfaceType);16 }17 }

运行结果

动态代理类图

等待上传中。

第三种:InterfaceProxyWithoutTarget

代码示例

1             { 2                 Console.WriteLine("\n*************InterfaceProxyWithoutTarget*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var animal = generator.CreateInterfaceProxyWithoutTarget
(new AnimalInterceptor()); 5 animal.Play(); 6 7 Console.WriteLine(animal.GetType()); 8 Console.WriteLine(animal.GetType().BaseType); 9 10 var compositeField = animal.GetType().GetField("__target");11 Console.WriteLine(compositeField);12 13 foreach (var interfaceType in animal.GetType().GetInterfaces())14 {15 Console.WriteLine(interfaceType);16 }17 }

运行结果

动态代理类图

等待上传中。

第四种:InterfaceProxyWithTarget

测试代码

1             { 2                 Console.WriteLine("\n*************InterfaceProxyWithTarget*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var animal = generator.CreateInterfaceProxyWithTarget
(new Dog(), new AnimalInterceptor()); 5 animal.Play(); 6 7 Console.WriteLine(animal.GetType()); 8 Console.WriteLine(animal.GetType().BaseType); 9 10 var compositeField = animal.GetType().GetField("__target");11 Console.WriteLine(compositeField);12 13 foreach (var interfaceType in animal.GetType().GetInterfaces())14 {15 Console.WriteLine(interfaceType);16 }17 }

运行结果

动态代理类图

等待上传中。

第五种:InterfaceProxyWithTargetInterface

测试代码

1             { 2                 Console.WriteLine("\n*************InterfaceProxyWithTargetInterface*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var animal = generator.CreateInterfaceProxyWithTargetInterface
(new Dog(), new AnimalInterceptor()); 5 animal.Play(); 6 7 Console.WriteLine(animal.GetType()); 8 Console.WriteLine(animal.GetType().BaseType); 9 10 var compositeField = animal.GetType().GetField("__target");11 Console.WriteLine(compositeField);12 13 foreach (var interfaceType in animal.GetType().GetInterfaces())14 {15 Console.WriteLine(interfaceType);16 }17 }

运行结果

动态代理类图

等待上传中。

1种Mixin模式

测试代码

1             { 2                 Console.WriteLine("\n*************Mixin*************\n"); 3                 var generator = new ProxyGenerator(); 4                 var options = new ProxyGenerationOptions(); 5                 options.AddMixinInstance(new RunAbility()); 6                 var animal = generator.CreateClassProxy
(options, new AnimalInterceptor()); 7 animal.Play(); 8 (animal as IRunable).Run(); 9 10 Console.WriteLine(animal.GetType());11 Console.WriteLine(animal.GetType().BaseType);12 13 var compositeField = animal.GetType().GetField("__target");14 Console.WriteLine(compositeField);15 16 foreach (var field in animal.GetType().GetFields())17 {18 if (field.Name.StartsWith("__mixin"))19 {20 Console.WriteLine(field);21 }22 }23 24 foreach (var interfaceType in animal.GetType().GetInterfaces())25 {26 Console.WriteLine(interfaceType);27 }28 }

运行结果

动态代理类图

等待上传中。

动态代理在DCI中的应用

参考链接:。

经常见到的动态代理场景

  1. ORM延时加载。
  2. AOP拦截(不全是使用的动态代理,有的使用透明代理或字节码增强,有的使用平台自带的管道过滤器,如:ASP.NET MVC的FilterAction)。
  3. WCF客户端代理。

备注

了解了这些模式后,自己开发一个简单的动态代理模式应该不是问题了,如果是C#语言,得学好Emit(不是那么简单),如果是Ruby的话,估计就非常Easy了,找个机会给出这两种语言的不同实现。

 

转载地址:http://uuhql.baihongyu.com/

你可能感兴趣的文章
如何永久的修改主机名
查看>>
NSSearchPathForDirectoriesInDomains用法(后台缓存)
查看>>
Jqurey 全选和全不选
查看>>
ELK日志收集平台部署
查看>>
软件公司员工辞职、人员流动大是必然
查看>>
勤快的love枫[ZJOI2007]
查看>>
Linux查看系统信息的一些命令及查看已安装软件包的命令
查看>>
poj1417 true liars(并查集 + DP)详解
查看>>
离散数学--二元关系总结
查看>>
HTML5 本地存储 localStorage、sessionStorage 的遍历、存储大小限制处理
查看>>
【leetcode】688. Knight Probability in Chessboard
查看>>
【leetcode】Maximum Product of Word Lengths
查看>>
C 工具库 GLib --- 提供多种高级的数据结构,如内存块、双向和单向链表、哈希表、动态字符串等...
查看>>
SQL中format()函数对应的格式
查看>>
svn command
查看>>
职业插画之路
查看>>
Java入门篇(五)——字符串/String类
查看>>
python 的StringIO
查看>>
第三个阶段事后诸葛亮
查看>>
java中的sql语句中如果有like怎么写
查看>>