引入
代理设计模式的原理:使用一个代理将对象包装起来,然后用该代理对象取代原始对象。任何对原始对象的调用都要通过代理。代理对象决定是否以及何时将方法调用转到原始对象上。
上一篇设计模式讲到的代理模式属于静态代理,特征是代理类和目标对象的类都是在编译期间确定下来,不利于程序的扩展。同时,每一个代理类只能为一个接口服务,这样一来程序开发中必然产生过多的代理。最好可以通过一个代理类完成全部的代理功能。
简介
动态代理是指客户通过代理类来调用其它对象的方法,并且是在程序运行时根据需要动态创建目标类的代理对象。
动态代理使用场合:
- 调试
- 远程方法调用
步骤
AOP
Aspect Oriented Programming(AOP),面向切面编程。
使用Proxy生成一个动态代理时,往往并不会凭空产生一个动态代理,这样没有太大的意义。通常都是为指定的目标对象生成动态代理。
这种动态代理在AOP中被称为AOP代理,AOP代理可代替目标对象,AOP代理包含了目标对象的全部方法。但AOP代理中的方法与目标对象的方法存在差异:AOP代理里的方法可以在执行目标方法之前、之后插入一些通用处理。
代码语言:javascript复制import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
// 接口
interface StudentInfoService{
void findInfo(String studentName);
void showRole();
}
// 被代理类
class StudentInfoServiceImpl implements StudentInfoService{
@Override
public void findInfo(String studentName) {
System.out.println("The name is " studentName);
}
@Override
public void showRole() {
System.out.println("班干部");
}
}
class MyHandler implements InvocationHandler {
private Object obj;
public void bind(Object obj) {
this.obj = obj;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("======调用日志方法" method.getName() "======");
Object returnVal = method.invoke(obj, args);
System.out.println("======日志方法" method.getName() "调用结束======");
return returnVal;
}
}
class AOPFactory{
// 动态创建创建一个代理类的对象
public static Object getProxyInstance(Object obj) {
MyHandler handler = new MyHandler();
handler.bind(obj);
return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), handler);
}
}
public class TestAOP {
public static void main(String[] args) {
// 创建一个被代理类的对象
StudentInfoServiceImpl serviceImpl = new StudentInfoServiceImpl();
// 动态获取一个代理类的对象
StudentInfoService studentInfoService = (StudentInfoService) AOPFactory.getProxyInstance(serviceImpl);
// 通过代理类的对象调用重写的抽象方法
studentInfoService.findInfo("runner");
studentInfoService.showRole();
}
}
代码语言:javascript复制运行结果:
======调用日志方法findInfo======
The name is runner
======日志方法findInfo调用结束======
======调用日志方法showRole======
班干部
======日志方法showRole调用结束======
总结
核心在于实现InvocationHandler接口和Proxy类的静态方法newProxyInstance,前者负责与被代理类关联(重写的invoke方法是关键),后者负责动态生成一个代理类。这两者再通过newProxyInstance的第三个参数关联起来,从而实现了动态代理。
AOP是动态代理的一种应用,其核心在于实现InvocationHandler的invoke方法,method.invoke(obj, args)是动态的,其前后可能有其他的硬编码。