不知各位是否还记得这两篇文章APP启动流程解析和AndroidHook告诉你如何启动未注册的Activity,这两篇文章中使用的技术基础都包含了代理模式,其中在文章中也说道
问许多工程师,设计模式用过哪些,相信很多人都会说单例模式、工厂模式等等,但是很少有人提及桥接模式、门面模式、解释器模式等甚至都没有听说过。
代理模式是结构型模式之
一,主要是将类和对象结合在一起解决特定的应用场景问题。对于Android工程师来说,我觉得了解并掌握代理模式是必要的,因为了解AndroidHook、AMS代理等插件化技术,是离不开代理模式的,这也是我一直觉得要有这篇文章的原因,如果你还不了解代理模式对Android开发者有什么用途,可移步至前言的两篇文章。
代理模式简单的说就是可以在不改被代理类代码的情况下,通过引入代理类来扩展功能。比如我们现在有一个登录注册类LoginAndRegist.java和一个登录方法一个注册方法。
那么我们现在有需求,为登录和注册添加相关日志,我们该怎么来实现呢,你可能说了,这不简单吗,直接在login和regist方法中直接再加两行打印不就行了吗?
那么我们该如何实现上面的功能呢,我们以为登录注册方法添加时间日志为例,首先有经验的工程师在写代码的时候,就应该知道,我们要遵循基于接口而非实现的设计原则,所以我们应把login和regist抽取出来,让LoginAndRegist类继承盖接口,如下所示:
为LoginAndRegist类创建代理类LoginAndRegistProxy,让代理类实现和原始类同样的接口,并调用原始类的方法,并在调用前打印当前时间戳,代码如下所示:
这样呢,我们就实现了不改变原始的情况下,为类方法添加日志的功能,但是呢,这种方法存在的问题我们上面也提到了
对于原始类并没有实现接口,并且我们无法修改的情况下,这种我们称为对外部类的扩展,外部类的扩展我们一般使用继承的方式去扩展,这种方式我们就不解释了。
第三个问题才是我们实际开发中需要首要解决的问题,所以为了解决静态代理文件过多的的问题,我们需要使用动态代理。
动态代理简单的说就是我们不需要事先为某个原始类编写代理类,而是在运行的时候,动态的创建代理类,然后将原始类替换为代理类。动态代理的魅力在Android中真的是非常非常大,如果你还不了解,一定要回头看我前言中提到的两篇文章。而在java中动态代理的基础是反射,如果你还不了解反射技术,请移步至我的这篇文章Java反射技术详解
动态代理,我们主要依赖的是newProxyInstance方法,该方法返回的是指定接口代理类的实例。
第二个参数interfaces是设置为对象class所实现的接口类型,第一个参数和第二个参数其实在业务上都是固定的,在这里就是UserInter对应的的classLoader和接口类型。
我们主要来看第三个参数InvocationHandler,它是一个实现了InvocationHandler接口的类对象
首先我们来定义一个MyInvocationHandler实现InvocationHandler
这里实现的invoke方法就是动态代理的核心,此外我们需要将代理类传进来,并在invoke方法中执行代理方法
当外部代理类调用某个方法的时候,其实就是在调用invoke中的method.invoke方法,args就是执行对应方法所需的参数,我们这里在方法前后分别加上日志。
上面我们已经说了,动态代理是通过newProxyInstance方法创建的,我们来看Test中如何使用
在Android中我们用Proxy.newProxyInstance生成的对象,直接替换掉原来的对象,这个技术就是听起来很高大上的Hook技术。
在业务系统中一些非功能性需求,比如:监控、统计、鉴权、事务、日志等。我们将这些附加功能与业务功能解耦,放到代理类中统一处理,这样可以与业务解耦并且做到职责明确划分。
除此之外代理模式还在RPC技术、AndroidHook、插件化等技术领域有着广泛的应用。