一. 代理模式
代理模式广泛的用于多种框架,如 Mybatis, Spring 等。有静态代理、动态代理两种。
1.1 静态代理
静态代理的核心主要是实现接口,在编译期实现的代理。代理类实现该接口,并在实现方法的时候包装方法,实现代理的目的。
// 接口类
public interface IUserDao {
void save();
}
// 实现类
public class UserDao implements IUserDao {
public void save() {
System.out.println("----已经保存数据!----");
}
}
// 代理类
public class UserDaoProxy implements IUserDao{
//接收保存目标对象
private IUserDao target;
public UserDaoProxy(IUserDao target){
this.target=target;
}
public void save() {
System.out.println("开始事务...");
target.save();//执行目标对象的方法
System.out.println("提交事务...");
}
}
1.2 动态代理
1.2.1 动态代理的原理
动态代理是在程序运行时期生成的代理对象。主要调用的是 Proxy#newProxyInstance() 方法,它可以在程序运行期间动态生成一个代理对象,用来代理实例对象。newProxyInstance 有三个参数:
- ClassLoader loader: 被代理类的类加载器;
- Class[] interfaces: 被代理类所实现的接口;
- InvocationHandler h: 主要内容,绑定被代理类的一个方法;
其中观察 newProxyInstance 方法源码可以得知,类加载器和接口主要是为了通过类加载器和接口获取这个被代理的 Class 对象,然后通过反射调用构造函数 new 一个实例对象。构建出对象之后,将 InvocationHandler 绑定到该对象上。所以 InvocationHandler 才是代理类主要的实现内容。
InvocationHandler 主要有三个参数:
- proxy: 被代理后的实例对象;
- method: 被代理的方法;
- args: 调用时的参数;
InvocationHandler 只有一个接口方法 invoke,在该接口方法的实现中,通常 method 的反射方法 method.invoke(proxy, args) 是核心内容,其余的增强方法包装着 method,这样就是实现了类的动态代理。
1.2.2 动态代理的应用场景
比如 Mybatis 在 sqlSession 获取 Mapper 的过程中,就是通过 MapperProxyFactory 构建动态代理类获取 Mapper 实体类(h 为继承了 InvocationHandler 的 MapperProxy)。
注:见《数据库》篇。
比如 Spring AOP 的实现中,接口类的切面是由 JDK 的动态代理实现的。
同理,Spring 事务的实现中,接口类的事务也是由 JDK 的动态代理实现的。