Java动态代理

概述

代理模式,是一种常用的设计模式。

在某些情况下,我们不希望或不能直接访问对象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();
}

// 不对外提供method3()
}

上面ClassA是委托类,ClassB是代理类,ClassB中的函数直接调用ClassA中相应的函数,并隐藏了ClassAmethod3()函数。

动态代理

代理类在程序运行前不存在,运行时由程序动态生成的代理方式称为动态代理。

动态代理的好处:可以方便对代理类的函数做统一或特殊处理,如记录所有函数的执行时间、所有函数执行前添加验证判断、对某个特殊函数进行特殊操作,而不用像静态代理方式那样需要修改每个函数。

实现动态代理的步骤:

  1. 新建委托类;
  2. 实现InvocationHandler接口,这是负责连接代理类和委托类的中间类必须实现的接口;
  3. 通过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;
}

/**
*
* @param proxy 表示通过Proxy.newProxyInstance()生成的代理类对象
* @param method 表示代理对象被调用的函数
* @param args 表示代理对象被调用的函数的参数
* @return
* @throws Throwable
*/
@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:是负责连接代理类和委托类的中间类必须实现的接口。调用代理对象的每个函数实际最终都是调用了InvocationHandlerinvoke函数。我们就可以在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

说明:

  1. 将委托类new OperateImpl()作为TimingInvocationHandler的构造参数创建handler对象;
  2. 通过Proxy.newProxyInstance(...)函数新建一个代理对象,代理类就是在这时候动态生成的;
  3. 调用代理对象的函数就会调用到handlerinvoke函数,而invoke函数中调用委托类对象相应的函数。
1
2
3
4
5
6
7
/**
*
* @param loader 类加载器
* @param interfaces 委托类的接口,生成代理类时需要实现这些接口
* @param handler handler是InvocationHandler的实现类对象,负责连接代理类和委托类的中间类
*/
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler handler)

参考链接

  1. 公共技术点之 Java 动态代理
  2. Java动态代理解析
  3. 十分钟理解Java之动态代理
  4. Java动态代理机制详解
文章目录
  1. 1. 概述
  2. 2. 静态代理
  3. 3. 动态代理
  4. 4. 参考链接
,