代理模式可分为静态代理和动态代理,什么是代理模式以及代理模式怎么使用?
代理模式
1. 代理模式的定义
为其他对象提供一种代理以控制对这个对象的访问。也称为“委托模式”。根据字节码的创建时机又分为静态代理和动态代理。
如果根据字节码的创建时机来分类,可以分为静态代理和动态代理:
静态代理:在程序运行前就已经存在代理类的字节码文件,代理类和真实主题角色的关系在运行前就确定了。
动态代理:源码是在程序运行期间由JVM根据反射等机制动态的生成,所以在运行前并不存在代理类的字节码文件。
2. 静态代理
普通代理:上层模块必须知道代理类的存在。访问真实主题角色的时候,要通过代理,因此要在上层模块创建代理对象。代理对象关联真实角,第一种方式:创建代理对象时,传一个真实对象的引用;第二种方式:在代理类内部实现。如果采用第二种关联的方式,在代理类创建时,由代理类决定创建一个真实角色,上层模块也并不需要知道真实角色的具体类。
强制代理:真实角色必须通过代理才能访问,并且真实角色的代理的产生,由真实对象决定(与普通代理恰好相反)
2.1 普通代理
1 | interface Subject { |
上层模块在创建代理时,并不用知道真实角色,但是要知道代理角色的”类”(通过new创建一个RealSubject对象,一定要知道类型)。简而言之,是将真实类对象传给代理类。
2.2 强制代理
对于普通代理来说,上层模块一定要知道代理的类型,但对于强制代理来说,强制代理的创建由真实角色决定,也就是说,通过真实角色找到代理,然后通过代理完成真实角色要完成的工作。
1 |
|
强制代理就是必须通过真实角色获得代理,然后由代理去完成真实角色的工作。除了强制代理需为接口增加获取代理的方法外,与普通代理不同的是强制代理的生成需要在真实角色的内部new出来,并将真是角色的this(本身对象)传给new出来的代理。
2.3 静态代理的缺点
虽然静态代理实现简单,且不侵入原代码,但是,当场景稍微复杂一些的时候,静态代理的缺点也会暴露出来。
当需要代理多个类的时候,由于代理对象要实现与目标对象一致的接口,有两种方式:
只维护一个代理类,由这个代理类实现多个接口,但是这样就导致代理类过于庞大
新建多个代理类,每个目标对象对应一个代理类,但是这样会产生过多的代理类
当接口需要增加、删除、修改方法的时候,目标对象与代理类都要同时修改,不易维护。
3. 动态代理
3.1 实现原理
- 设计动态代理类(
DynamicProxy
)时,不需要显式实现与目标对象类(RealSubject
)相同的接口,而是将这种实现推迟到程序运行时由JVM
来实现
- 即:在使用时再创建动态代理类 & 实例
- 静态代理则是在代理类实现时就指定与目标对象类(
RealSubject
)相同的接口
- 通过
Java
反射机制的method.invoke()
,通过调用动态代理类对象方法,从而自动调用目标对象的方法
3.2 步骤详解
步骤1: 声明目标对象的抽象接口
Subject.java
1 | public interface Subject { |
步骤2: 声明目标对象类
Buyer1.java
1 | // 小成,真正的想买Mac的对象 = 目标对象 = 被代理的对象 |
Buyer2.java
1 | // 小何,真正的想买iPhone的对象 = 目标对象 = 被代理的对象 |
步骤3: 声明代理类
DynamicProxy.java
1 | public class DynamicProxy implements InvocationHandler { |
步骤4: 通过动态代理对象,调用目标对象的方法
Main.java
1 | public static void main(String args[]) { |
3.3 优点
- 只需要1个动态代理类就可以解决创建多个静态代理的问题,避免重复、多余代码
- 更强的灵活性
- 设计动态代理类(
DynamicProxy
)时,不需要显式实现与目标对象类(RealSubject
)相同的接口,而是将这种实现推迟到程序运行时由JVM
来实现- 在使用时(调用目标对象方法时)才会动态创建动态代理类 & 实例,不需要事先实例化
3.4 缺点
- 效率低
相比静态代理中 直接调用目标对象方法,动态代理则需要先通过Java
反射机制 从而 间接调用目标对象方法 - 应用场景局限
因为 Java 的单继承特性(每个代理类都继承了 Proxy 类),即只能针对接口 创建 代理类,不能针对类 创建代理类
即只能动态代理 实现了接口的类
- 本文作者: zicair
- 本文链接: https://zicair.github.io/2020/07/25/动态代理模式/
- 版权声明: 本博客所有文章除特别声明外,均采用 MIT 许可协议。转载请注明出处!