首先说下为什么要有动态代理呢?设计模式中,有一种模式叫代理模式,为什么要使用动态代理其实和为什么要使用设计模式一样,请读者自行思考。

代理通常有静态代理,动态代理,其中静态代理是通过持有目标对象引用,然后对目标对象的方法进行增强,可以说是一种硬编码。jdk动态代理也需要持有目标对象的引用,在目标对象的方法的调用处进行增强,但这部分被封在处理器InvocationHandler中,通过反射机制生成Proxy实例,并传入处理器。----先来看下使用jdk动态代理实现方式:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class DynamicProxy implements InvocationHandler {
  private Object target;
  public DynamicProxy(Object object){
   this.target = object;
  }
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("before......");
    Object result = method.invoke(target,args);
    System.out.println("after......");
    return result;
  }
  @SuppressWarnings("unchecked")
  public <T> T getProxy(){
    return (T)Proxy.newProxyInstance(
         target.getClass().getClassLoader(),
         target.getClass().getInterfaces(),
          this);
  }
}

动态是相对于静态代理而言,jvm执行java类,需要先将class文件加载到jvm中,java是不允许在运行时修改class文件的,所以要想使用动态代理,需要动态生成class文件,加载到jvm中。通过设置系统属性,可以在运行时看到jdk动态生成的class文件。

System.getProperties().setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
在生成的类文件中,可以看到代理类继承自Proxy并实现了业务自定义的接口,下面是动态生成的代理类部分代码:
public final class $Proxy0 extends Proxy implements IHello{
 public final String hello(String var1) throws  {
    return (String)super.h.invoke(this, m3, new Object[]{var1});
  }
  static {
    m0 = Class.forName("java.lang.Object").getMethod("hashCode");
   m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
   m2 = Class.forName("java.lang.Object").getMethod("toString");
   m3 = Class.forName("com.proxy.IHello").getMethod("hello", Class.forName("java.lang.String"));
  }
}
可以看到jdk动态代理生成的代码,将目标对象接口的所有方法进行了代理,目标方法的执行就变成了处理器的invoke,在invoke中已经对目标对象中进行了增强。
其实思路是朴素的,就好比你不想出去吃饭点了外卖,外卖就相当于你的代理,帮你送饭,但是吃饭还是需要你自己来,外卖平台已经知道你需要的增强就是有人送饭,然后动态给你指定一个外卖骑手......----
下面看下jdk动态代理为了生成代理类是如何操作的,首先查找或生成设计好的代理class:

由ProxyClassFactory在给定类加载器和接口数组的条件下,生成、定义、并返回代理class。

最后将class字节码加载到jvm,在生成代理类的函数中可以看到是否生成class文件的布尔标记saveGeneratedFiles:

jdk动态代理无处不在,大名鼎的spring AOP默认使用的就是jdk动态代理,对于没有接口实现的类才使用cglib代理。


©著作权归作者所有:来自51CTO博客作者mb5fdb0a4002420的原创作品,如需转载,请注明出处,否则将追究法律责任

更多相关文章

  1. Vue3动态添加路由及生成菜单
  2. 如何让多端口网站用一个nginx进行反向代理实际场景分析
  3. MAC使用Safari浏览器仿真IE
  4. Linux下安装配置Cntlm代理
  5. 选择屏幕动态显示BLOCK title
  6. 掌握Mybatis动态映射,我可是下了功夫的
  7. 学习LINUX的第17天
  8. nginx反向代理和负载均衡策略实战案例
  9. 常用的字符串方法介绍以及使用事件代理实现一个简单的留言板

随机推荐

  1. [java混淆器问题] 如何让web项目的class
  2. JSP访问类实例
  3. eclipse写javaweb工程时文件目录为WebCon
  4. 启动非java虚拟机方式下运行matlab
  5. 转贴:JavaScript实现Ajax请求简单示例
  6. java之finally的用法
  7. 域对象/服务和业务逻辑层
  8. java常量的定义
  9. JAVA-全局变量与局部变量-继承-封装-(是三
  10. python 日志简单使用