设计模式总结之一——代理模式

一. 代理模式

代理模式广泛的用于多种框架,如 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 的动态代理实现的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值