转自:https://blog.csdn.net/educast/article/details/6565447#
动态代理的原理

原理其实很简单,就是在运行时生成新的对象,姑且叫做T,并使T继承自需要代理的原对象,调用过程实际是调用了新的对象T.

通过对T中方法或属性等,添加些自定义的操作,从而实现对原对象访问的封装.

动态代理实现(利用castle)

castle的动态代理需要下面几步

  1. 自定义一个拦截器,必须实现 IInterceptor 接口
  2. 使用 ProxyGenerator 对象创建代理对象,对象中包含很多方法
  3. 对原对象的所有操作,都使用代理对象代替
  4. 在拦截器的方法中,加入自定义的操作,比如 记录参数调用日志,异常记录等.

简单拦截器实现代码:  SampleInterceptor.cs

  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. using Castle.Core.Interceptor;
  11. using System.Reflection;
  12. /// <summary>
  13. /// 拦截器示例
  14. /// </summary>
  15. public class SampleInterceptor : IInterceptor
  16. {
  17. public SampleInterceptor()
  18. {
  19. //
  20. //TODO: 在此处添加构造函数逻辑
  21. //
  22. }
  23. public void Intercept(IInvocation invocation)
  24. {
  25. output("开始进入拦截器");
  26. MethodInfo concreteMethod = invocation.GetConcreteMethod();
  27. if (!invocation.MethodInvocationTarget.IsAbstract)
  28. {
  29. output("开始执行 " + concreteMethod.Name);
  30. //执行原对象中的方法
  31. invocation.Proceed();
  32. output("执行结果 " + invocation.ReturnValue);
  33. }
  34. output("执行完毕");
  35. }
  36. private void output(string Message)
  37. {
  38. HttpContext.Current.Response.Write(Message + "<br>");
  39. }
  40. }
  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. using Castle.Core.Interceptor;
  11. using System.Reflection;
  12. /// <summary>
  13. /// 拦截器示例
  14. /// </summary>
  15. public class SampleInterceptor : IInterceptor
  16. {
  17. public SampleInterceptor()
  18. {
  19. //
  20. //TODO: 在此处添加构造函数逻辑
  21. //
  22. }
  23. public void Intercept(IInvocation invocation)
  24. {
  25. output("开始进入拦截器");
  26. MethodInfo concreteMethod = invocation.GetConcreteMethod();
  27. if (!invocation.MethodInvocationTarget.IsAbstract)
  28. {
  29. output("开始执行 " + concreteMethod.Name);
  30. //执行原对象中的方法
  31. invocation.Proceed();
  32. output("执行结果 " + invocation.ReturnValue);
  33. }
  34. output("执行完毕");
  35. }
  36. private void output(string Message)
  37. {
  38. HttpContext.Current.Response.Write(Message + "<br>");
  39. }
  40. }

示例中使用的接口 : IPerson.cs

  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. /// <summary>
  11. ///IPerson 的摘要说明
  12. /// </summary>
  13. public interface IPerson
  14. {
  15. /// <summary>
  16. /// 姓名
  17. /// </summary>
  18. string Name { get; }
  19. /// <summary>
  20. /// 地址
  21. /// </summary>
  22. string Address { get; }
  23. /// <summary>
  24. /// 正在做什么
  25. /// </summary>
  26. /// <returns></returns>
  27. string Doing();
  28. }
  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. /// <summary>
  11. ///IPerson 的摘要说明
  12. /// </summary>
  13. public interface IPerson
  14. {
  15. /// <summary>
  16. /// 姓名
  17. /// </summary>
  18. string Name { get; }
  19. /// <summary>
  20. /// 地址
  21. /// </summary>
  22. string Address { get; }
  23. /// <summary>
  24. /// 正在做什么
  25. /// </summary>
  26. /// <returns></returns>
  27. string Doing();
  28. }

对接口的实现:Person.cs

  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. /// <summary>
  11. ///Person 的摘要说明
  12. /// </summary>
  13. public class Person : IPerson
  14. {
  15. public Person()
  16. {
  17. //
  18. //TODO: 在此处添加构造函数逻辑
  19. //
  20. }
  21. #region IPerson 成员
  22. public string Name
  23. {
  24. get { return "我是花生米"; }
  25. }
  26. public string Address
  27. {
  28. get { return "我住在 http://pignut-wang.iteye.com/ "; }
  29. }
  30. public string Doing()
  31. {
  32. return "我正在写blog";
  33. }
  34. #endregion
  35. }
  1. using System;
  2. using System.Data;
  3. using System.Configuration;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. /// <summary>
  11. ///Person 的摘要说明
  12. /// </summary>
  13. public class Person : IPerson
  14. {
  15. public Person()
  16. {
  17. //
  18. //TODO: 在此处添加构造函数逻辑
  19. //
  20. }
  21. #region IPerson 成员
  22. public string Name
  23. {
  24. get { return "我是花生米"; }
  25. }
  26. public string Address
  27. {
  28. get { return "我住在 http://pignut-wang.iteye.com/ "; }
  29. }
  30. public string Doing()
  31. {
  32. return "我正在写blog";
  33. }
  34. #endregion
  35. }

所有要使用到的对象都准备好了,下面就是调用的代码

  1. using System;
  2. using System.Configuration;
  3. using System.Data;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. using Castle.DynamicProxy;
  11. public partial class _Default : System.Web.UI.Page
  12. {
  13. protected void Page_Load(object sender, EventArgs e)
  14. {
  15. //创建拦截器对象
  16. SampleInterceptor Interceptor = new SampleInterceptor();
  17. //给person类生成代理
  18. ProxyGenerator Generator = new ProxyGenerator();
  19. IPerson p = Generator.CreateInterfaceProxyWithTarget<IPerson>(new Person(), Interceptor);
  20. //执行方法看效果
  21. p.Doing();
  22. }
  23. }
  1. using System;
  2. using System.Configuration;
  3. using System.Data;
  4. using System.Web;
  5. using System.Web.Security;
  6. using System.Web.UI;
  7. using System.Web.UI.HtmlControls;
  8. using System.Web.UI.WebControls;
  9. using System.Web.UI.WebControls.WebParts;
  10. using Castle.DynamicProxy;
  11. public partial class _Default : System.Web.UI.Page
  12. {
  13. protected void Page_Load(object sender, EventArgs e)
  14. {
  15. //创建拦截器对象
  16. SampleInterceptor Interceptor = new SampleInterceptor();
  17. //给person类生成代理
  18. ProxyGenerator Generator = new ProxyGenerator();
  19. IPerson p = Generator.CreateInterfaceProxyWithTarget<IPerson>(new Person(), Interceptor);
  20. //执行方法看效果
  21. p.Doing();
  22. }
  23. }

执行的效果就是在页面上输出4句话,如下

开始进入拦截器
开始执行 Doing
执行结果 我正在写blog
执行完毕

castle动态代理的使用的更多相关文章

  1. Castle动态代理拦截

    比如现在有一个方法,进行积分奖励 PointAdd 在不改变原来方法的基础上,增加积分奖励的日志 using Castle.DynamicProxy; public class AuditTraceI ...

  2. Aop之使用Castle动态代理实现对方法的拦截

    using System; using System.Linq; using Castle.DynamicProxy; namespace AopTest { class AopTest { stat ...

  3. AOP之Castle DynamicProxy 动态代理

    这里主要介绍使用castle这个动态代理,在.net一些开源的框架里可以找到它的影子,就连微软的rchard也是使用这个进行方法拦截等可以基于这个进行方法拦截,在这个方面PostSharp算是比较好用 ...

  4. castle之动态代理

    动态代理 DynamicProxy,这里说的动态代理是直接使用Castle.net 中提供的,并非自己实现的,因为别人写的很好,拿着用就行了. 动态代理的工作模式: 一般我们获取一个类型的实例都是通过 ...

  5. Castle Windsor 的动态代理类如何获取实际类型

    问题 在实际开发过程当中我们可能会针对某些类型使用动态代理技术(AOP),注入了一些拦截器进行处理,但是一旦某个类型被动态代理了,那么就会生成一个代理类.这个时候在该类内部使用 GetType() 方 ...

  6. ASP.NET Core搭建多层网站架构【9.2-使用Castle.Core实现动态代理拦截器】

    2020/01/31, ASP.NET Core 3.1, VS2019, Autofac.Extras.DynamicProxy 4.5.0, Castle.Core.AsyncIntercepto ...

  7. 不用Unity库,利用.NET动态代理自己实现AOP

    AOP意为面向切面的程序设计,主要表现为对不同的代码逻辑进行隔离,从而降低不同业务逻辑之间的耦合性,AOP又理解为“横切”,可以在不改变原有实现的情况下,对代码进行拦截和扩展,如果原有设计像一个瓶子, ...

  8. Autofac高级用法之动态代理

    前言 Autofac的DynamicProxy来自老牌的Castle项目.DynamicProxy(以下称为动态代理)起作用主要是为我们的类生成一个代理类,这个代理类可以在我们调用原本类的方法之前,调 ...

  9. 5.动态代理AOP实现-DynamicProxy模式

    通过动态代理模式Interceptor实现在RegUser()方法本身业务前后加上一些自己的功能,如:PreProceed和PostProceed,即不修改UserProcessor类又能增加新功能 ...

随机推荐

  1. C-Lodop回调函数的触发

    高版本的火狐和谷歌不再支持np插件之后,Lodop公司推出了C-Lodop,解决了这些浏览器不能用Lodop插件方式打印的问题,相比较Lodop插件,C-Lodop由于是以服务的形式出现,返回值不能直 ...

  2. BZOJ4551[Tjoi2016&Heoi2016]树——dfs序+线段树/树链剖分+线段树

    题目描述 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为1),有以下 两种操作:1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标记,其他结点均 ...

  3. MT【208】埃尔米特恒等式

    设$S=\sum\limits_{k=1}^{+\infty}[\dfrac{116+3^{k-1}}{3^k}]\\T=\sum\limits_{k=1}^{+\infty}[\dfrac{116+ ...

  4. 12 Zabbix Item类型之Zabbix JMX类型

    点击返回:自学Zabbix之路 12 Zabbix Item类型之Zabbix JMX类型 JMX 全称是Java Management Extensions,即Java管理扩展.Java程序会开放一 ...

  5. 自学Zabbix3.12-动作Action

    点击返回:自学Zabbix之路 点击返回:自学Zabbix4.0之路 点击返回:自学zabbix集锦 自学Zabbix3.12-动作Action介绍 1. 动作action 在配置好监控项和触发器之后 ...

  6. BZOJ3118 Orz the MST 【单纯形 + 生成树】

    题目链接 BZOJ3118 题解 少有的单纯形好题啊 我们先抽离出生成树 生成树中的边只可能减,其它边只可能加 对于不在生成树的边,其权值一定要比生成树中其端点之间的路径上所有的边都大 然后就是一个最 ...

  7. 洛谷 P3102 [USACO14FEB]秘密代码Secret Code 解题报告

    P3102 [USACO14FEB]秘密代码Secret Code 题目描述 Farmer John has secret message that he wants to hide from his ...

  8. Bean和Spirng模块

    容纳Bean 在Spring中,应用对象生存于Spring容器中,如图所示,Spring容器可以创建.装载.配置这些Bean,并且可以管理它们的生命周期. Spring的容器实现 Bean工厂(org ...

  9. P1856 矩形周长

    哇!这小破题坑了我好久. 扫描线+线段树 这题数据范围小,没离散化.真要离散化我还搞不好呢. 具体的看这个博客吧. 主要是这个坑爹的c,len把我搞了,其他的还好. 代码: #include < ...

  10. ASP: Response 对象 错误 'ASP 0251 : 80004005' 解决办法

    Response 对象 错误 'ASP 0251 : 80004005' 超过响应缓冲区限制 这种情况一般是因为需要输出的网页内容太大了,由于asp在输入内容到客户的浏览器上之前,会把需要输出的全部内 ...