随笔 - 597  文章 - 4  评论 - 445  阅读 - 424万

设计模式之代理模式

1、代理模式,就是接口 + 真实实现类 + 代理类。其中真实实现类和代理类都是要实现接口的,实例化的时候使用代理类。Spring AOP要做的是生成一个代理类来替换掉真实实现的类以对外提供服务。   

  在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。代理模式为其他对象提供一种代理以控制对这个对象的访问。

  在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。

2、首先,创建一个支付的接口。

复制代码
 1 package com.example.aop;
 2 
 3 /**
 4  * 支付的接口
 5  */
 6 public interface Payment {
 7 
 8     /**
 9      *
10      */
11     public void pay();
12 
13 }
复制代码

然后,创建一个真实的支付类,实现支付接口,用于顾客的实际支付,但是顾客不关心怎么支付。

复制代码
 1 package com.example.aop;
 2 
 3 /**
 4  * 真实的支付操作
 5  */
 6 public class RealPayment implements Payment {
 7 
 8     @Override
 9     public void pay() {
10         // 用户只关心支付功能
11         System.out.println("作为顾客,我只关注支付功能!");
12     }
13 }
复制代码

最后,创建一个代理对象,代理对象,代理顾客去进行支付操作。 

复制代码
 1 package com.example.aop;
 2 
 3 /**
 4  * 背后的转账和取钱功能需要支付宝来完成,它相当于是一个代理对象
 5  * <p>
 6  * 支付宝作为一个代理存在的
 7  */
 8 public class AliPayment implements Payment {
 9 
10     // 支付宝需要调用客户的支付
11     private Payment payment;
12 
13     /**
14      * 含参构造函数
15      *
16      * @param payment
17      */
18     public AliPayment(Payment payment) {
19         this.payment = payment;
20     }
21 
22     /**
23      * 支付之前,支付宝去银行卡取款
24      */
25     public void beforePay() {
26         System.out.println("支付宝去银行卡里面取款.");
27     }
28 
29     @Override
30     public void pay() {
31         // 支付前的操作
32         this.beforePay();
33 
34         // 客户支付操作
35         payment.pay();
36 
37         // 支付后的操作
38         this.afterPay();
39     }
40 
41     /**
42      * 支付之后,支付宝将钱打给谁
43      */
44     public void afterPay() {
45         System.out.println("支付宝将钱给商家.");
46     }
47 
48 
49     public static void main(String[] args) {
50         // 以支付宝为代理进行RealPayment支付操作
51         Payment proxy = new AliPayment(new RealPayment());
52         proxy.pay();
53 
54     }
55 
56 }
复制代码

 

3、AOP的实现,JdkProxy 和 Cglib。

  1)、由AopProxyFactory根据AdvisedSupport对象的配置来决定。
  2)、Spring默认的策略如果目标类是接口,则用JDKProxy来实现,否则用后者Cglib来生成代理。
  3)、JDKProxy的核心,InvocationHandler接口和Proxy类。JdkProxy通过反射来接收被代理的类,并且要求被代理的类必须实现一个接口,JdkProxy动态代理的核心是InvocationHandler接口和Proxy类。
  4)、如果目标类没有实现接口,以继承的方式动态生成目标类的代理,就需要使用Cglib代理了。Cglib动态代理m目标类,是一个代码生成的类库,可以在运行的时候动态生成某个类的子类,是通过修改字节码的方式来实现的。如果某个类标记为final,那么这个类是无法使用Cglib来做动态代理的。
  5)、JdkProxy,是通过Java的内部反射机制来实现的。
  6)、Cglib的底层是借助ASM实现的。ASM是一种能够操作字节码的框架。总的来说,反射机制在生成类的过程中比较高效,ASM在生成类之后的执行过程中比较高效,实际上,我们可以通过j将ASM生成的类进行缓存,这样可以解决ASM生成类过程比较低效的问题。

 

4、Spring里面的代理模式的实现。

  1)、真实实现类的逻辑包含在getBean方法里面。getBean方法用于查找或者实例化容器中的bean,这也是Spring AOP只能作用于Spring容器中的bean的原因,对于不是Spring容器管理的对象,Spring AOP是无能为力的。
  2)、getBean方法返回的实际上是Proxy的实例。
  3)、代理类Proxy实例是Spring采用JdkProxy或者Cglib动态生成的。

 

posted on   别先生  阅读(431)  评论(0编辑  收藏  举报
编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

点击右上角即可分享
微信分享提示