java动态代理

动态代理

什么是代理

   代理是一种软件设计模式,这种设计模式不直接访问被代理对象,而访问被代理对象的方法,详尽的解释可参考《java设计模式之禅》,里面的解释还是很通俗的。给个《java设计模式之禅》下载地址:https://pan.baidu.com/s/1GdFmZSx67HjKl_OhkwjqNg

   在JDK中提供了实现动态代理模式的机制,cglib也是一个用于实现动态代理的框架,在这里我介绍jdk自带的动态代理机制是如何使用的。先上代码再慢慢解释:

// 定义一个接口,接口中只有一个抽象方法,这个方法便是将要被代理的方法
public interface Subject {
  String test(String string);
}

// 定义 一个类实现这个方法,方法里写上自己的逻辑。
public class SubjectImpl implements Subject {

  @Override
  public String test(String string) {
      String test=string+string;
      return test;
  }
}
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
// 定义一个代理类,这个代理必须要实现invocaHandler 接口,用以指定这个类为一个代理类
public class MyProxy implements InvocationHandler {
    private Object target;
    public MyProxy( Object target){
        this.target=target;
    }
    // 实现 invoke方法,这个方法将是后面代码中实际执行的方法
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("开始执行动态代理");
        Object result=method.invoke(target,args);
        return  result;
    }
    // Proxy类为反射机制中的一个类,用于获得代理对象
    public  T getProxy(){
      /* newProxyInstance方法的参数解释:
      ** 被代理对象的类加载器:target.getClass().getClassLoader()
      ** 被代理的方法:target.getClass().getInterfaces()
      **代理对象:this
      */
        return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
    }
}
public static void main(String[] args) {
        // 创建一个被代理对象
       Subject subject=new SubjectImpl();
       // 创建一个代理对象
       Subject proxy=new MyProxy(subject).getProxy();
       // 调用代理的方法
       System.out.println(proxy.test("test"));
   }

输出结果:

开始执行动态代理
testtest

   我们知道spring 的AOP是通过动态代理实现的,现在让我们好好分析一下动态代理,示例中定义了一个接口 Subject,一个继承接口的SubjectImpl类,一个实现了InvocationHandler的MyProxy类,并调用了Proxy.newProxyInstance方法。Subject定义了将要被代理执行的方法,SubjectImpl是被代理的类(雇主),MyProxy类是代理执行的类(跑腿的),它的invoke(Object proxy, Method method, Object[] args)方法便是实际被执行的方法,它的第一个参数proxy作用:

  • 可以使用反射获取代理对象的信息(也就是proxy.getClass().getName())
  • 可以将代理对象返回以进行连续调用,这就是proxy存在的目的,因为this并不是代理对象(MyProxy虽然是代理的类,但代理对象是 Proxy.newProxyInstance方法生成的。)。

method 是被代理的方法类型对象,args是方法的参数数组。通过Proxy.newProxyInstance生成代理类后就可以执行其中的代理方法了。

   如果我们直接执行SubjectImpl.test()方法则只返回一个字符串,但使用动态代理我们可以在方法执行前和执行后加上自己的逻辑,这样大大提高了代码的复用性;想一想,如果你写了一堆方法,方法里很多代码是一样的,这样的代码是不是很丑?好,现在你把重复的代码单独抽出来做一个方法,但这样你的每个方法都被写死了,和那个公共的方法耦合在一起,这样很不灵活。如果我突然想一部分方法的公共方法是a(),一部分方法的公共方法是b(),那改起来很麻烦,扩展性很差。使用动态代理就很好的解决了这个问题,被代理对象可以任意指定,代理的逻辑可以任意实现,二者互相独立互不影响,并且可以由客户端任意进行组合,这就是所谓的动态。

你可能感兴趣的:(java动态代理)