概述
代理模式,是一种常用的设计模式。
在某些情况下,我们不希望或不能直接访问对象A,而是通过访问一个中介对象B,由B去访问A达成目的,这种方式就是代理。
对象A所属的类称为委托类
,也被称为被代理类
,对象B所属的类称为代理类
。
根据程序运行前代理类是否存在,可以将代理分为静态代理
和动态代理
。
静态代理
代理类在程序运行前已经存在的代理方式称为静态代理。
由开发人员编写或是编译器生成代理类的方式都属于静态代理。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| class ClassA { public void method1() {} public void method2() {} public void method3() {} }
public class ClassB { private ClassA a; public ClassB(ClassA a) { this.a = a; } public void method1() { a.method1(); } public void method2() { a.method2(); } }
|
上面ClassA
是委托类,ClassB
是代理类,ClassB
中的函数直接调用ClassA
中相应的函数,并隐藏了ClassA
的method3()
函数。
动态代理
代理类在程序运行前不存在,运行时由程序动态生成的代理方式称为动态代理。
动态代理的好处:可以方便对代理类的函数做统一或特殊处理,如记录所有函数的执行时间、所有函数执行前添加验证判断、对某个特殊函数进行特殊操作,而不用像静态代理方式那样需要修改每个函数。
实现动态代理的步骤:
- 新建委托类;
- 实现
InvocationHandler
接口,这是负责连接代理类和委托类的中间类必须实现的接口;
- 通过
Proxy
类创建代理类对象。
接下来我们通过一个实例来演示动态代理的使用。如果要统计某个类所有函数的执行时间,传统的方式是在类的每个函数前打点统计,使用动态代理可以对这一操作进行统一处理。
Step1. 新建委托类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
| public interface Operate { void method1(); void method2(); void method3(); }
public class OperateImpl implements Operate { private static final String TAG = "DynamicProxy";
@Override public void method1() { Log.i(TAG, "Invoke method1"); sleep(100); }
@Override public void method2() { Log.i(TAG, "Invoke method2"); sleep(87); }
@Override public void method3() { Log.i(TAG, "Invoke method3"); sleep(28); }
private static void sleep(long millSeconds) { try { Thread.sleep(millSeconds); } catch (InterruptedException e) { e.printStackTrace(); } } }
|
Step2. 实现InvocationHandler接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| public class TimingInvocationHandler implements InvocationHandler { private static final String TAG = "DynamicProxy";
private Object target;
public TimingInvocationHandler(Object target) { this.target = target; }
@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { long start = System.currentTimeMillis(); Object obj = method.invoke(target, objects); Log.i(TAG, method.getName() + " cost time is:" + (System.currentTimeMillis() - start)); return obj; } }
|
InvocationHandler
:是负责连接代理类和委托类的中间类必须实现的接口。调用代理对象的每个函数实际最终都是调用了InvocationHandler
的invoke
函数。我们就可以在invoke
函数中添加开始结束计时,其中还调用了委托类对象target
的相应函数,这样便完成了统计执行时间的需求。
Step3. 通过Proxy类静态函数动态生成代理对象
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| public class MainActivity extends AppCompatActivity {
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TimingInvocationHandler handler = new TimingInvocationHandler(new OperateImpl()); Operate operate = (Operate) Proxy.newProxyInstance(Operate.class.getClassLoader(), new Class[]{Operate.class}, handler);
operate.method1();
operate.method2();
operate.method3(); } }
|
执行结果:
1 2 3 4 5 6
| I/DynamicProxy: Invoke method1 I/DynamicProxy: method1 cost time is:100 I/DynamicProxy: Invoke method2 I/DynamicProxy: method2 cost time is:88 I/DynamicProxy: Invoke method3 I/DynamicProxy: method3 cost time is:28
|
说明:
- 将委托类
new OperateImpl()
作为TimingInvocationHandler
的构造参数创建handler
对象;
- 通过
Proxy.newProxyInstance(...)
函数新建一个代理对象,代理类就是在这时候动态生成的;
- 调用代理对象的函数就会调用到
handler
的invoke
函数,而invoke
函数中调用委托类对象相应的函数。
1 2 3 4 5 6 7
|
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler handler)
|
参考链接
- 公共技术点之 Java 动态代理
- Java动态代理解析
- 十分钟理解Java之动态代理
- Java动态代理机制详解