一.代理设计模式
代理模式主要对我们的方法前后实现增强
二.代理模式的应用场景
- 打印日志
- Aop
- 事务Aop
- MyBatis中maper接口
- 全局捕获异常
- Lcn,seata,分表分框架shadingjdbc代理数据源
- 自定义注解生效(反射技术+Aop)
- Rpc远程调用技术,代理设计模式
三.代理模式的实现方式
- 静态代理
- 需要人工的手写代理类
- 动态代理
- JDK动态代理(反射拼接Java代码-实现接口)
- CGLIB动态代理(拼接字节码-继承)
四.代理模式优缺点
- 优点
- 实现扩展功能,对我们方法实现增强、安全性、冗余性提高代码复用机制。
- 缺点
- 生成非常多代理的class文件
五.Jdk动态代理模式底层实现
package cn.tedu.proxy.jdk;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class JdkInvocationHandler implements InvocationHandler {
/**
* 目标对象(需要被代理对象)
*/
private Object target;
public JdkInvocationHandler(Object target) {
this.target = target;
}
@Override
/**
* proxy JavaJdk动态生成代理类对象$proxy0
* method 目标方法 反射获取到的
* args 目标方法需要传递的参数
*
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Jdk动态代理方式开始:");
// 执行目标方法 根据接口的方法 反射技术执行目标对象的对应的方法
Object result = method.invoke(target, args);
System.out.println("Jdk动态代理方式结束:");
/***
* Method method 类型接口的方法 采用java反射机制执行我们的对应目标对象的方法
*/
return result;
}
/**
* 通过Jdk动态代理反射技术生成代理对象 调用代理类对象的方法的时候会走InvocationHandlerinvoke方法
*
* @param <T>
* @return
*/
public <T> T getProxy() {
return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(), this);
}
// public <T> T getProxy() {
// return (T) new $Proxy0(this);
// }
}
手写JavaJdk动态代理实现思路:
1.根据接口的信息动态拼接代理类的源代码$Proxy.java
2. $Proxy.java编译为 $Proxy.class
3.基于类加载器动态读取 $Proxy.class到内存中