Java动态代理讲解和示例

上一篇 / 下一篇  2023-02-24 14:02:55

  Java动态代理是一种在运行时创建代理类的机制,动态代理可以在不修改源代码的情况下,在运行时为某个接口动态生成实现类,并且可以拦截接口中的方法调用,从而实现一些特殊的功能。
  Java 动态代理底层原理是基于反射机制实现的,其中最重要的是 InvocationHandler 接口,它定义了一个 invoke() 方法,用于实现对代理类中各个方法的调用,在 invoke() 方法中,可以实现对真实角色的调用,并进行扩展,实现动态代理的效果。
  动态代理的实现步骤:
  1. 创建一个实现 InvocationHandler 接口的类,它必须实现 invoke() 方法;
  2. 创建被代理的真实对象;
  3. 通过 Proxy 类的 newProxyInstance() 方法创建代理对象,它需要参数:
  (1)ClassLoader:类加载器,它是用于加载代理对象字节码的,和被代理对象使用相同的类加载器;
  (2)Class[]:字节码数组,它是用于让代理对象和被代理对象有相同方法;
  (3)InvocationHandler:它是调用处理器,执行代理对象的方法时,会触发该对象的 invoke() 方法;
  4. 通过代理对象调用目标方法,实际上会转到 invoke() 方法中,在 invoke() 方法中,可以进行预处理、调用后处理等工作。
  反射上一篇文章讲过了,这里就是反射的一个应用,这个最常见的场景就是给代码加日志,例如在执行某个函数前将请求体加入到日志中,执行后将结果加入到日志中。这样可以在不改变原来代码的基础上来实现。接下来就使用上一篇反射的例子继续扩展。
  先写一个Person的接口:
  package ReflectTest;
  public interface PersonInterface {
      void printName();
      void printAge();
  }
  写他的实现:
  package ReflectTest;
  public class Person implements PersonInterface{
      private String name = "xiaoming";
      private String age = "12";
      @Override
      public void printName()
      {
          System.out.println(name);
      }
      @Override
      public void printAge()
      {
          System.out.println(age);
      }
  }
  然后写动态代理类:
  package ReflectTest;
  import java.lang.reflect.InvocationHandler;
  import java.lang.reflect.Method;
  import java.lang.reflect.Proxy;
  public class ProxyHandler implements InvocationHandler {
      private Object target;
      public ProxyHandler(Object target) {
          this.target = target;
      }
      public Object bind() {
          return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(), this);
      }
      @Override
      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
          System.out.println("执行函数前加入日志");
          Object result = method.invoke(target, args);
          System.out.println("执行函数后加入日志");
          return result;
      }
  }
  最后测试一下:
  package ReflectTest;
  public class main {
      public static void main(String[] args) {
          ProxyHandler proxyHandler = new ProxyHandler(new Person());
          PersonInterface person = (PersonInterface)proxyHandler.bind();
          person.printName();
      }
  }
  我们看看结果在执行函数前后执行了其他操作:
  SpringAOP其实原理就类似这种。

TAG: 软件开发 Java java

 

评分:0

我来说两句

Open Toolbar