JDK动态代理

什么是代理 所谓代理,就是将要执行的动作,请一个第三方的代替执行。一个现实中的例子,比如,有一个人要做手术时,需要家属签字,那么,这个过程中,家属就相当于是病人的代理,家属会代替病人签字,出院的时候,家属也可以代替病人办理出院。而在手术的过程中,医生需要用到的一些药物,也需要家属签字,家属也可以选择不签,在这个过程中,家属这个代理是可以完全控制的。
在java中,即一个真实的实现类,它的行为会用一个它的代理类去执行,代理类在执行真实的类的时候,可以在具体的方法执行前后增加一些额外的逻辑操作。比如方法执行前的初始化,或者方法执行后的清理等工作。代理可以分为静态代理和动态代理。
静态代理,就是事先在java代码中写好代理类,实现代理类的逻辑。动态代理则是在代码运行的过程中,动态的为真实类添加代理类。从上面的定义不难理解,静态代理是事先编写好的,如果后期有功能变化,则需要同步修改代理类,而动态代理则是在运行时动态生成,则不需要如此麻烦的操作。
【JDK动态代理】java中的动态代理,比较常用的实现方式有两种,一种是JDK实现的动态代理,一种是CGLIB实现的动态代理。前者主要基于接口实现,后者基于继承实现。本文主要学习JDK的动态代理。
JDK动态代理 JDK动态代理主要涉及两个类:java.lang.reflect.Proxy、java.lang.reflect.InvocationHandler。在使用的过程中将会用到Proxy类中的newProxyInstance方法以及实现InvocationHandler接口。还是以病人动手术为例。
首先,需要定义一个病人的接口:

/** * 定义一个病人的接口,规范人的动作 */ public interface Patient { /** * 病人可以签字 */ void sign(); /** * 办理入院 */ void toHospital(); /** * 办理出院 */ void leaveHospital(); }

实现一个需要动手术的病人:
/** * 一个需要动手术的病人 */ public class SurgeryPatient implements Patient { @Override public void sign() { System.out.println("手术中,签字"); }@Override public void toHospital() { System.out.println("手术前,办理入院"); }@Override public void leaveHospital() { System.out.println("手术后,办理出院"); } }

实现病人的代理:
/** * 实现一个病人的代理,为手术病人办理入院,签字,出院等。 */ public class PatientProxy implements InvocationHandler { private Object object; public PatientProxy(Object o) { this.object = o; }@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("代理开始为病人执行相关动作。。。"); Object invoke = method.invoke(object, args); System.out.println("代理为病人执行完动作"); return invoke; } }

调用逻辑:
/** * 测试类 */ public class App { public static void main(String[] args) { // 首先需要一个动手术的病人 Patient surgeryPatient = new SurgeryPatient(); // 为这个病人创建一个代理 PatientProxy patientProxy = new PatientProxy(surgeryPatient); Patient patient = (Patient) Proxy.newProxyInstance(patientProxy.getClass().getClassLoader(), new Class[]{Patient.class}, patientProxy); // 办理入院 patient.toHospital(); // 手术时签字 patient.sign(); //办理出院 patient.leaveHospital(); } }

运行结果:
代理开始为病人执行相关动作。。。 手术前,办理入院 代理为病人执行完动作 代理开始为病人执行相关动作。。。 手术中,签字 代理为病人执行完动作 代理开始为病人执行相关动作。。。 手术后,办理出院 代理为病人执行完动作

通过这个例子,不难想到Spring中AOP的使用。此外,日志的收集以及数据库连接中,连接前的初始化,连接后的资源释放等工作,是否都可以使用动态代理处理呢?

    推荐阅读