介绍
代理模式(Proxy Pattern) 也称为委托模式,是结构性设计模式。
定义
为其他对象提供一种代理以控制对这个对象的访问
使用场景
当无法或不想直接访问某个对象或者访问某个对象存在困难时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,委托对象与代理对象需要实现相同的接口
不想直接访问或者访问某个对象困难,主要指某个类的方法访问比较复杂不是直接调一下方法即可,这时候可以用一个代理类来做这部分工作,我们使用时直接调用代理类的方法即可。
角色介绍
Subject 抽象主题类,该类的职责是声明真实主题与代理的共同接口方法,该类既可以是一个抽象类也可以是一个接口
RealSubject 真实主题类,该类也被称为被委托类或被代理类,该类定义了代理所表示的真实对象,由其执行具体的业务逻辑方法,而客户端类则通过代理类间接地调用真实主题类中的定义的方法
ProxySubject 代理类,该类也称为委托类或代理类,该类持有一个对真实主题类的引用,在其所实现的接口方法中调用真是主题类中相应的接口方法执行,一次起到代理的作用。
Client 客户端类,即使用代理类的类型。
代理模式其实就是一种委托机制,真是对象将方法的执行委托给代理类。其实我们的代理类完全可以代理多个被代理类,具体代理的到底是哪个类,这要看代理类中持有的实际对象类型。
静态代理
代理者的代码由程序员或自己通过一些自动化工具生产固定代码再对其编译,也就是我们的代码运行前 代理类 的 class 编译文件就已经存在,这样的就是静态代理
动态代理
通过反射机制动态地生成代理者的对象,也就是说我们在 code 阶段压根不知道代理谁,代理谁将在执行阶段决定。Java 给我们提供了一个便捷的动态代理接口 InvocationHandler ,实现该接口需要重写其调用方法 invoke。
public interface InvocationHandler {
Object invoke(Object var1, Method var2, Object[] var3) throws Throwable;
}
public class DynamicProxy implements InvocationHandler {
private Object object;
public DynamicProxy(Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
return method.invoke(object, args);
}
}
DynamicProxy proxy = new DynamicProxy(xiaoming);
ClassLoader loader = xiaomin.getClass().getClassLoader();
ILawsuit layer = Proxy.newProxyInstance(loader,new Class[]{ILawsuit.class},proxy)
这里通过 invoke 方法来调用具体的被代理方法,也就是真实的方法。动态代理可以使我们的代码逻辑更简洁。
在 Dynamic 类中,Object 就是被代理对象。
使用时,首先要获取被代理类的 ClassLoader 对象,再有被代理类构造动态代理类对象,之后由 Proxy 的 newProxyInstance 方法,传入被代理类的 ClassLoader、Class 数组 以及动态代理类 DynamicProxy 的对象,返回结果为被代理类的接口类型的代理对象。
动态代理通过一个代理类来代理 N 多个被代理类,其实质是对代理者和被代理者进行解耦,使两者之间没有直接的耦合关系。相对而言静态代理则只能为给定接口下的实现类做代理,如果接口不同那么就需要定义不同的代理类,较为复杂,但是静态代理更符合面向对象原则。
Android 源码中的代理模式
IActivityManager 为接口类型,其中定义了一系列方法,相当于代理模式中的抽象主题。
ActivityMangerProxy 实现了 IActivityManager 接口,相当于代理模式中的代理者
ActivityManagerNative 也实现了 IActivityManager,但是是个抽象类,其子类 ActivityManagerService 实现了它的抽象方法,相当于代理模式中的被代理者
工作过程
Android 中管理与维护 Activity 的类是 AcitivityManager 类,相当于代理模式中的 Client 类,以其中的 getAppTasks 方法为例
调用 ActivityManagerNative.getDefault().getAppTasks() 方法,ActivityManagerNative.getDefault() 其实就是通过 ServiceManager.getService("activity") 方法得到 AMS,然后通过 AMP 代理 AMS,返回 AMP,到这里相当于代理模式中由被代理者构造代理者。
AMP 的 getAppTasks 方法,则是通过 APM 代理者操作 AMS 被代理者的 getAppTasks 方法。
在这里是通过 Binder 完成 AMP 和 AMS 直接的通信,是一个跨进程的过程,加这层代理也主要是为了解决跨进程工作。
开发
通知栏 Notification 的开发
通知栏有多种,高度为 64dp 的,高度为 256dp 的,还有悬浮类型的,不过这几种都有 send 和 cancle 两个通用方法,所以可以将通知类抽象出一个 Nofity 抽象接口,三种通知都实现 Notify 接口,实现各自的 send 和 cancel 方法, 再通过 NofityProxy 来实现 Nofity ,其中定义一个 Nofity 类型的成员,在构造 NotifyProxy 构造时传入 Nofity 类型的对象,在 NotifyProxy 的 send 和 cancel 方法中,调用 Nofity 类成员的对应方法。