JDK动态代理和cglib动态代理

冷不防 2022-04-22 07:56 449阅读 0赞

转载:https://www.jianshu.com/p/1712ef4f2717

回顾一下JDK动态代理和cglib动态代理,转自我的BLOG:

http://zeusjava.com/2015/10/13/jdk-dynamic-agent-and-cglib/

闲言少叙,先来说一下什么是代理模式,我们去一个新的地方总是要先找地方住,但是我们人生地不熟的掌握的资源不多,这时候一般会找中介,中介对房源很熟悉,很快就能为你找到合适的房子,这时候,中介就是一个代理,你就相当于是一个委托方。

下面是设计模式中的代理:

代理模式

代理模式是常用的java设计模式,他的特征是代理类委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

按照代理的创建时期,代理类可以分为两种:

静态代理

  1. 由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。

动态代理

  1. 在程序运行时,运用反射机制动态创建而成。 动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。

JDK动态代理

先来看下JDK源码中InvocationHandler中invoke()方法

  1. public Object invoke(Object proxy, Method method, Object[] args)
  2. throws Throwable;

JDK源码中Proxy类的代码:

  1. public static Object newProxyInstance(ClassLoader loader,
  2. Class<?>[] interfaces,
  3. InvocationHandler h)
  4. throws IllegalArgumentException
  5. {
  6. if (h == null) {
  7. throw new NullPointerException();
  8. }
  9. /*
  10. * Look up or generate the designated proxy class.
  11. */
  12. Class<?> cl = getProxyClass(loader, interfaces);
  13. /*
  14. * Invoke its constructor with the designated invocation handler.
  15. */
  16. try {
  17. Constructor cons = cl.getConstructor(constructorParams);
  18. return cons.newInstance(new Object[] { h });
  19. } catch (NoSuchMethodException e) {
  20. throw new InternalError(e.toString());
  21. } catch (IllegalAccessException e) {
  22. throw new InternalError(e.toString());
  23. } catch (InstantiationException e) {
  24. throw new InternalError(e.toString());
  25. } catch (InvocationTargetException e) {
  26. throw new InternalError(e.toString());
  27. }
  28. }

参数说明:

  1. ClassLoader loader:类加载器
  2. Class<?>[] interfaces:得到全部的接口
  3. InvocationHandler h:得到InvocationHandler接口的子类实例

PS:类加载器

在Proxy类中的newProxyInstance()方法中需要一个ClassLoader类的实例,ClassLoader实际上对应的是类加载器,
在Java中主要有一下三种类加载器:

  1. Booststrap ClassLoader:此加载器采用C++编写,一般开发中是看不到的;
  2. Extendsion ClassLoader:用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类;
  3. AppClassLoader:(默认)加载classpath指定的类,是最常使用的是一种加载器。

JDK动态代理实现步骤

实现InvocationHandler接口

获得代理对象

  1. public Object getInstance(Object target){
  2. return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
  3. }

回调函数

  1. @Override
  2. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  3. Object result = null;
  4. System.out.println("jdk 动态代理 begin...");
  5. result = method.invoke(target,args);
  6. System.out.println("jdk 动态代理 end...");
  7. return result;
  8. }

JDK动态代理缺点:

  1. 只能对实现了接口的类进行,没有实现接口的类不能使用JDK动态代理。

cglib动态代理

JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
cglib实现动态代理的方法和JDK动态代理类似

实现MethodInterceptor接口

获得代理对象

  1. public Object getInstance(Object target){
  2. this.target = target;
  3. Enhancer enhancer = new Enhancer();
  4. enhancer.setSuperclass(this.target.getClass());
  5. //设置回调方法
  6. enhancer.setCallback(this);
  7. //创建代理对象
  8. return enhancer.create();
  9. }

设置回调方法

  1. @Override
  2. public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
  3. System.out.println("UserFacadeProxy.intercept begin");
  4. methodProxy.invokeSuper(o,objects);
  5. System.out.println("UserFacadeProxy.intercept end");
  6. return null;
  7. }

Spring AOP原理

java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
SpringAOP动态代理策略是:

  1. 1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
  2. 2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
  3. 3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

发表评论

表情:
评论列表 (有 0 条评论,449人围观)

还没有评论,来说两句吧...

相关阅读

    相关 JDK动态代理CGLib动态代理

        JDK 动态代理是 JVM 根据传进来的对象,动态的创建对象的代理对象并返回。     CGLib 动态代理比较复杂,它是通过继承的方式来实现类的代理。 JDK