模式之动态代理模式
1.代理模式(Proxy Pattern):静态代理 - 最易懂的设计模式解析
2.代理模式(Proxy Pattern):动态代理 - 最易懂的设计模式解析(转载)
1分钟全面了解“设计模式”
单例模式(Singleton) - 最易懂的设计模式解析
简单工厂模式(SimpleFactoryPattern)- 最易懂的设计模式解析
工厂方法模式(Factory Method)- 最易懂的设计模式解析
抽象工厂模式(Abstract Factory)- 最易懂的设计模式解析
策略模式(Strategy Pattern)- 最易懂的设计模式解析
适配器模式(Adapter Pattern)- 最易懂的设计模式解析
代理模式(Proxy Pattern)- 最易懂的设计模式解析
模板方法模式(Template Method) - 最易懂的设计模式解析
建造者模式(Builder Pattern)- 最易懂的设计模式解析
外观模式(Facade Pattern) - 最易懂的设计模式解析
1.代理模式(Proxy Pattern):静态代理 - 最易懂的设计模式解析
1. 介绍
1.1 定义:
给目标对象提供一个代理对象,并由代理对象控制对目标对象的引用
代理对象:起到中介作用,连接客户端和目标对象
例子:电脑桌面的快捷方式。电脑对某个程序提供一个快捷方式(代理对象),快捷方式连接客户端和程序,客户端通过操作快捷方式就可以操作那个程序
1.2 主要作用:
通过引入代理对象的方式来间接访问目标对象
1.3 解决的问题:
防止直接访问目标对象给系统带来的不必要复杂性。
2. 模式原理
2.1 UML类图 & 组成
2.2 实例讲解
接下来我用一个实例来对代理模式进行更深一步的介绍。
a. 实例概况
背景:小成希望买一台最新的顶配Mac电脑
冲突:国内还没上,只有美国才有
解决方案:寻找代购进行购买
代购(代理对象) 代替 我(真实对象) 去买Mac(间接访问的操作)
b. 使用步骤
步骤1: 创建抽象对象接口(Subject):声明你(真实对象)需要让代购(代理对象)帮忙做的事(买Mac)
public interface Subject {
public void buyMac();
}
步骤2: 创建真实对象类(RealSubject),即”我“
public class RealSubject implement Subject{
@Override
public void buyMac() {
System.out.println(”买一台Mac“);
}
}
步骤3:创建代理对象类(Proxy),即”代购“,并通过代理类创建真实对象实例并访问其方法
public class Proxy implements Subject{
@Override
public void buyMac{
//引用并创建真实对象实例,即”我“
RealSubject realSubject = new RealSubject();
//调用真实对象的方法,进行代理购买Mac
realSubject.buyMac();
//代理对象额外做的操作
this.WrapMac();
}
public void WrapMac(){
System.out.println(”用盒子包装好Mac“);
}
}
步骤4:客户端调用
public class ProxyPattern {
public static void main(String[] args){
Subject proxy = new Proxy();
proxy.buyMac();
}
}
结果输出
买一台Mac
用盒子包装好Mac
通过上述这个常见的生活例子,相信你已经完全明白了代理模式原理了!!
3. 优缺点
3.1 优点
协调调用者和被调用者,降低了系统的耦合度
代理对象作为客户端和目标对象之间的中介,起到了保护目标对象的作用
3.2 缺点
由于在客户端和真实主题之间增加了代理对象,因此会造成请求的处理速度变慢;
实现代理模式需要额外的工作(有些代理模式的实现非常复杂),从而增加了系统实现的复杂度。
4. 应用场景
3. 具体应用
接下来,我将用1个具体实例来对 动态代理模式 进行更深一步的介绍
3.1 实例概况
背景:小成 希望买一台最新的顶配 Mac 电脑;小何希望买一台 iPhone
冲突:国内还没上,只有美国才有
解决方案:寻找一个代购一起进行购买
即1个代购(动态代理对象)同时 代替 小成 & 小何(目标对象) 去买Mac(间接访问的操作)
该代购是代购任何商品 = 什么人有什么需求就会去代购任何东西(动态代理)
3.2 使用步骤
声明 调用处理器类
声明目标对象类的抽象接口
声明目标对象类
通过动态代理对象,调用目标对象的方法
3.3 步骤详解
步骤1: 声明 调用处理器类
DynamicProxy.java
<-- 作用 --> // 1. 生成 动态代理对象 // 2. 指定 代理对象运行目标对象方法时需要完成的 具体任务 // 注:需实现InvocationHandler接口 = 调用处理器 接口 // 所以称为 调用处理器类 public class DynamicProxy implements InvocationHandler { // 声明代理对象 // 作用:绑定关系,即关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke() private Object ProxyObject; public Object newProxyInstance(Object ProxyObject){ this.ProxyObject =ProxyObject; return Proxy.newProxyInstance(ProxyObject.getClass().getClassLoader(), ProxyObject.getClass().getInterfaces(),this); // Proxy类 = 动态代理类的主类 // Proxy.newProxyInstance()作用:根据指定的类装载器、一组接口 & 调用处理器 生成动态代理类实例,并最终返回 // 参数说明: // 参数1:指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器 // 参数2:指定目标对象的实现接口 // 即要给目标对象提供一组什么接口。若提供了一组接口给它,那么该代理对象就默认实现了该接口,这样就能调用这组接口中的方法 // 参数3:指定InvocationHandler对象。即动态代理对象在调用方法时,会关联到哪个InvocationHandler对象 } // 复写InvocationHandler接口的invoke() // 动态代理对象调用目标对象的任何方法前,都会调用调用处理器类的invoke() @Override public Object invoke(Object proxy, Method method, Object[] args) // 参数说明: // 参数1:动态代理对象(即哪个动态代理对象调用了method() // 参数2:目标对象被调用的方法 // 参数3:指定被调用方法的参数 throws Throwable { System.out.println("代购出门了"); Object result = null; // 通过Java反射机制调用目标对象方法 result = method.invoke(ProxyObject, args); return result; } } 步骤2: 声明目标对象的抽象接口 public interface Subject { // 定义目标对象的接口方法 // 代购物品 public void buy(); } 步骤3: 声明目标对象类 // 小成,真正的想买Mac的对象 = 目标对象 = 被代理的对象 // 实现抽象目标对象的接口 public class Buyer1 implements Subject { @Override public void buybuybuy() { System.out.println("小成要买Mac"); } } // 小何,真正的想买iPhone的对象 = 目标对象 = 被代理的对象 // 实现抽象目标对象的接口 public class Buyer2 implements Subject { @Override public void buybuybuy() { System.out.println("小何要买iPhone"); } } 步骤4: 通过动态代理对象,调用目标对象的方法 public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 1. 创建调用处理器类对象 DynamicProxy DynamicProxy = new DynamicProxy(); // 2. 创建目标对象对象 Buyer1 mBuyer1 = new Buyer1(); // 3. 创建动态代理类 & 对象:通过调用处理器类对象newProxyInstance() // 传入上述目标对象对象 Subject Buyer1_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer1); // 4. 通过调用动态代理对象方法从而调用目标对象方法 // 实际上是调用了invoke(),再通过invoke()里的反射机制调用目标对象的方法 Buyer1_DynamicProxy.buybuybuy(); // 以上代购为小成代购Mac // 以下是代购为小何代购iPhone Buyer2 mBuyer2 = new Buyer2(); Subject Buyer2_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer2); Buyer2_DynamicProxy.buybuybuy(); } }
3.4 测试结果
3.5 Demo地址
Carson_Ho的Github地址:动态代理DynamicProxy
4. 源码分析
-
在经过上面的实例后,你是否会对以下问题好奇:
- 动态代理类 及其对象实例是如何生成的?
- 如何通过调用动态代理对象方法,从而调用目标对象方法?
-
下面,我们顺着 步骤4:目标对象 通过 动态代理对象调用方法的使用 来进行动态代理模式的源码分析
// 步骤4:通过动态代理对象,调用目标对象的方法
此处有两个重要的源码分析点:
Subject Buyer1_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer1);//关注1
Buyer1_DynamicProxy.buybuybuy(); //关注2
- 关注1:创建动态代理类 & 对象:通过调用处理器类对象
newProxyInstance()
解决的问题是:动态代理类 及其对象实例是如何生成的?
- 关注2:通过调用动态代理对象方法从而调用目标对象方法
解决的问题是:如何通过调用动态代理对象方法,从而调用目标对象方法?
下面,我们将主要分析这两处源码。
4.1 (关注1)创建动态代理类 & 对象:通过调用处理器类对象newProxyInstance()
// 使用代码
Subject Buyer1_DynamicProxy = (Subject) DynamicProxy.newProxyInstance(mBuyer1);
- 即,动态代理类 及其对象实例是如何生成的?
- 下面,我们直接进入
DynamicProxy.newProxyInstance()
<-- 关注1:调用处理器 类的newProxyInstance() -->
// 即步骤1中实现的类:DynamicProxy
// 作用:
// 1. 生成 动态代理对象
// 2. 指定 代理对象运行目标对象方法时需要完成的 具体任务
// 注:需实现InvocationHandler接口 = 调用处理器 接口
public class DynamicProxy implements InvocationHandler {
// 声明代理对象
private Object ProxyObject;
public Object newProxyInstance(Object ProxyObject){
this.ProxyObject =ProxyObject;
return Proxy.newProxyInstance(ProxyObject.getClass().getClassLoader(),
ProxyObject.getClass().getInterfaces(),this);
// Proxy.newProxyInstance()作用:根据指定的类装载器、一组接口 & 调用处理器 生成动态代理类实例,并最终返回
// ->>关注2
}
// 以下暂时忽略,下文会详细介绍
// 复写InvocationHandler接口的invoke()
// 动态代理对象调用目标对象的任何方法前,都会调用调用处理器类的invoke()
@Override
public Object invoke(Object proxy, Method method, Object[] args)
// 参数说明:
// 参数1:动态代理对象(即哪个动态代理对象调用了method()
// 参数2:目标对象被调用的方法
// 参数3:指定被调用方法的参数
throws Throwable {
System.out.println("代购出门了");
Object result = null;
// 通过Java反射机制调用目标对象方法
result = method.invoke(ProxyObject, args);
return result;
}
// 至此,关注1分析完毕,跳出
}
<-- 关注2:newProxyInstance()源码解析-->
// 作用:根据指定的类装载器、一组接口 & 调用处理器 生成动态代理类及其对象实例,并最终返回
public static Object newProxyInstance(ClassLoader loader, Class<?>[]interfaces,InvocationHandler h) throws IllegalArgumentException {
// 参数说明:
// 参数1:指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器
// 参数2:指定目标对象的实现接口
// 即要给目标对象提供一组什么接口。若提供了一组接口给它,那么该代理对象就默认实现了该接口,这样就能调用这组接口中的方法
// 参数3:指定InvocationHandler对象。即动态代理对象在调用方法时,会关联到哪个InvocationHandler对象
...
// 仅贴出核心代码
// 1. 通过 为Proxy类指定类加载器对象 & 一组interface,从而创建动态代理类
// >>关注3
Class cl = getProxyClass(loader, interfaces);
// 2. 通过反射机制获取动态代理类的构造函数,其参数类型是调用处理器接口类型
Constructor cons = cl.getConstructor(constructorParams);
// 3. 通过动态代理类的构造函数 创建 代理类实例(传入调用处理器对象
return (Object) cons.newInstance(new Object[] { h });
// 特别注意
// 1. 动态代理类继承 Proxy 类 & 并实现了在Proxy.newProxyInstance()中提供的一系列接口(接口数组)
// 2. Proxy 类中有一个映射表
// 映射关系为:(<ClassLoader>,(<Interfaces>,<ProxyClass>) )
// 即:1级key = 类加载器,根据1级key 得到 2级key = 接口数组
// 因此:1类加载器对象 + 1接口数组 = 确定了一个代理类实例
...
// 回到调用关注2的原处
}
<-- 关注3:getProxyClass()源码分析 -->
// 作用:创建动态代理类
public static Class<?> getProxyClass(ClassLoader loader,Class<?>... interfaces) throws IllegalArgumentException {
...
// 仅贴出关键代码
<-- 1. 将目标类所实现的接口加载到内存中 -->
// 遍历目标类所实现的接口
for (int i = 0; i < interfaces.length; i++) {
// 获取目标类实现的接口名称
String interfaceName = interfaces[i].getName();
Class interfaceClass = null;
try {
// 加载目标类实现的接口到内存中
interfaceClass = Class.forName(interfaceName, false, loader);
} catch (ClassNotFoundException e) {
}
if (interfaceClass != interfaces[i]) {
throw new IllegalArgumentException(
interfaces[i] + " is not visible from class loader");
}
}
<-- 2. 生成动态代理类 -->
// 根据传入的接口 & 代理对象 创建动态代理类的字节码
byte