Spring之代理

代理模式的概念:

二十三种设计模式中的一种,属于结构型模式。它的作用就是通过提供一个代理类,让我们在调用目标方法的时候,不再是直接对目标方法进行调用,而是通过代理类间接调用。让不属于目标方法核心逻辑的代码从目标方法中剥离出来——解耦。调用目标方法时先调用代理对象的方法,减少对目标方法的调用和打扰,同时让附加功能能够集中在一起也有利于统一维护。

使用前:

 

使用后:

静态代理类:

public class CalculatorStaticProxy implements Calculator {
 
  // 将被代理的目标对象声明为成员变量
  private CalculatorImp target;  //CalculatorImp为Calculator的具体实现类,是核心代码 
  public CalculatorStaticProxy(Calculator target) {
    this.target = target;
 }
 
  @Override
  public int add(int i, int j) {
 
    // 附加功能由代理类中的代理方法来实现
    System.out.println("[日志] add 方法开始了,参数是:" + i + "," + j);
 
    // 通过目标对象来实现核心业务逻辑
    int addResult = target.add(i, j);
 
    System.out.println("[日志] add 方法结束了,结果是:" + addResult);
    return addResult;
 }
}

静态代理确实实现了解耦,但是由于代码都写死了,完全不具备任何的灵活性。就拿日志功能来
说,将来其他地方也需要附加日志,那还得再声明更多个静态代理类,那就产生了大量重复的代
码,日志功能还是分散的,没有统一管理。
提出进一步的需求:将日志功能集中到一个代理类中,将来有任何日志需求,都通过这一个代理
类来实现。这就需要使用动态代理技术了。 

 动态代理:

public class ProxyFactory {
  private Object target;
  public ProxyFactory(Object target) {
    this.target = target;
 }
  public Object getProxy(){
    /**
    * newProxyInstance():创建一个代理实例
    * 其中有三个参数:
    * 1、classLoader:加载动态生成的代理类的类加载器
    * 2、interfaces:目标对象实现的所有接口的class对象所组成的数组
    * 3、invocationHandler:设置代理对象实现目标对象方法的过程,即代理类中如何重写接
口中的抽象方法
    */
    ClassLoader classLoader = target.getClass().getClassLoader();
    Class<?>[] interfaces = target.getClass().getInterfaces();
    InvocationHandler invocationHandler = new InvocationHandler() {
      @Override
      public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
        /**
        * proxy:代理对象
        * method:代理对象需要实现的方法,即其中需要重写的方法
        * args:method所对应方法的参数
        */
Object result = null;
        try {
          System.out.println("[动态代理][日志] "+method.getName()+",参
数:"+ Arrays.toString(args));
          result = method.invoke(target, args);
          System.out.println("[动态代理][日志] "+method.getName()+",结
果:"+ result);
       } 
 }      catch (Exception e) {
          e.printStackTrace();
          System.out.println("[动态代理][日志] "+method.getName()+",异
常:"+e.getMessage());
       } finally {
          System.out.println("[动态代理][日志] "+method.getName()+",方法
执行完毕");
       }
        return result;
     }
   };
    return Proxy.newProxyInstance(classLoader, interfaces,
invocationHandler);  

}

 测试:

@Test
public void testDynamicProxy(){
  ProxyFactory factory = new ProxyFactory(new CalculatorLogImpl());
  Calculator proxy = (Calculator) factory.getProxy();
  proxy.div(1,0);
  //proxy.div(1,1);
}

 

 

 

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值