概述
代理模式,是一种常用的设计模式。 在某些情况下,我们不希望或不能直接访问对象A,而是通过访问一个中介对象B,由B去访问A达成目的,这种方式就是代理。 对象A所属的类称为
委托类
,也被称为被代理类
,对象B所属的类称为代理类
。 根据程序运行前代理类是否存在,可以将代理分为静态代理
和动态代理
。
代理类在程序运行前已经存在的代理方式称为静态代理。 由开发人员编写或是编译器生成代理类的方式都属于静态代理。
上面ClassA
是委托类,ClassB
是代理类,ClassB
中的函数直接调用ClassA
中相应的函数,并隐藏了ClassA
的method3()
函数。
代理类在程序运行前不存在,运行时由程序动态生成的代理方式称为动态代理。
动态代理的好处:可以方便对代理类的函数做统一或特殊处理,如记录所有函数的执行时间、所有函数执行前添加验证判断、对某个特殊函数进行特殊操作,而不用像静态代理方式那样需要修改每个函数。
实现动态代理的步骤:
InvocationHandler
接口,这是负责连接代理类和委托类的中间类必须实现的接口;Proxy
类创建代理类对象。接下来我们通过一个实例来演示动态代理的使用。如果要统计某个类所有函数的执行时间,传统的方式是在类的每个函数前打点统计,使用动态代理可以对这一操作进行统一处理。
Step1. 新建委托类
Step2. 实现InvocationHandler接口
InvocationHandler
:是负责连接代理类和委托类的中间类必须实现的接口。调用代理对象的每个函数实际最终都是调用了InvocationHandler
的invoke
函数。我们就可以在invoke
函数中添加开始结束计时,其中还调用了委托类对象target
的相应函数,这样便完成了统计执行时间的需求。
Step3. 通过Proxy类静态函数动态生成代理对象
执行结果:
说明:
new OperateImpl()
作为TimingInvocationHandler
的构造参数创建handler
对象;Proxy.newProxyInstance(...)
函数新建一个代理对象,代理类就是在这时候动态生成的;handler
的invoke
函数,而invoke
函数中调用委托类对象相应的函数。