Java两种方式实现动态代理

 更新时间:2020-10-10 04:02:15   作者:佚名   我要评论(0)

一、JDK动态代理
Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口、目标接口的类加载器以及 Invocation

一、JDK动态代理

Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口、目标接口的类加载器以及 InvocationHandler 便可为目标接口生成代理类及代理对象。我们称这个Java技术为:动态代理

@CallerSensitive
public static Object newProxyInstance(ClassLoader loader,
                   Class<?>[] interfaces,
                   InvocationHandler h)
  throws IllegalArgumentException
{
//...
}

在 Java 中规定,要想产生一个对象的代理对象,那么这个对象必须要有一个接口,因此 interfaces 必须是一个接口。

在动态代理技术里,由于不管用户调用代理对象的什么方法,都是调用开发人员编写的 InvocationHandler 的 invoke 方法(这相当于 invoke 方法拦截到了代理对象的方法调用)。

因此 JDK 动态代理的整体流程为:

  1. 实现 InvocationHandler,用来处理对象拦截后的逻辑。(该对象必须是接口,或者父类是接口)
  2. 使用 Proxy.newProxyInstance 产生代理对象。

以下是一个用 JDK 动态代码实现 AOP 的具体例子:

1.目标(Target)类

public interface UserService {
  void eat();
}
public class UserServiceImpl implements UserService {
  @Override
  public void eat() {
    System.out.println("吃东西");
  }
}

2.切面(Aspect)类

public class MyAspect {
  /**
   * 前置通知
   */
  public void before() {
    System.out.print("先洗手再");
  }
}

3. 织入(Weaving)过程

/**
 * 产生代理对象的工厂类
 */
public class MyFactoryBean {

  private MyFactoryBean() {
  }
  
  public static UserService getInstance() {
    // target : 目标类
    final UserService userService = new UserServiceImpl();
    // Aspect : 切面类
    final MyAspect myAspect = new MyAspect();
    // Weaving : 织入,也就是产生代理的过程
    UserService proxyInstance = (UserService) Proxy.newProxyInstance(UserService.class.getClassLoader(),
        new Class[]{UserService.class}, (Object proxy, Method method, Object[] args) -> {
          // 模拟切点 - pointcut
          if ("eat".equals(method.getName())) {
            myAspect.before();
          }
          return method.invoke(userService, args);
        });
    return proxyInstance;
  }
}
  public static void main(String[] args) {
    UserService userService = MyFactoryBean.getInstance();
    // 先洗手再吃东西
    userService.eat();
  }

想想看,这其实跟我们平常使用的 AOP 已经很相似了,Spring 里面定义了前置通知(@Before)、异常通知(@AfterThrowing)等等,Spring 只是换成了甄别这些注解来选择什么时候调用通知方法,另外,Spring 还通过切点表达式来选择目标类和切入点。

二、CGLIB动态代理

CGLIB 动态代理需要引入第三方的库,它通过修改代理对象生成子类的方式来实现调用拦截,代理对象不需要实现接口,但是代理类不能是 final,代理的方法也不能是 final。

/**
 * 产生代理对象的工厂类
 */
public class MyFactoryBean {

  private MyFactoryBean() {
  }

  public static UserService getInstance() {
    // target : 目标类
    final UserService userService = new UserServiceImpl();
    // Aspect : 切面类
    final MyAspect myAspect = new MyAspect();
    // Weaving : 织入,也就是产生代理的过程
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(userService.getClass());
    enhancer.setUseCache(false);
    enhancer.setCallback(new MethodInterceptor() {
      @Override
      public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        // 模拟 pointcut-切点
        if ("eat".equals(method.getName())) {
          myAspect.before();
        }
        return methodProxy.invokeSuper(o, objects);
      }
    });
    return (UserService) enhancer.create();
  }

  public static void main(String[] args) {
    UserService proxyInstance = MyFactoryBean.getInstance();
    // 先洗手再吃东西
    proxyInstance.eat();
  }
}

三、总结

在 JDK 中实现动态代理时,要求代理类必须是接口或继承接口的类,因为 JDK 最后生成的 proxy class 其实就是实现了代理类所代理的接口并且继承了 java 中的 Proxy 类(继承 Proxy 类是为了判断该类是否为代理类),通过反射找到接口的方法,调用 InvocationHandler的invoke 方法实现拦截。

CGLIB 字节码增强是JDK动态代理的一个很好的补充, CGLIB 中最后生成的 proxy class 是一个继承代理类所代理的 class,通过重写被代理类中的非 final 的方法实现代理。

总结为:

  • JDK 动态代理:代理类必须是接口或继承接口的类。
  • CGLIB 字节码增强: 代理类不能是 final,代理的方法也不能是 final(继承限制) 。

关于在 Spring 的 AOP 中采用何种代理手段,我们不强加限制的话,会根据类是否有接口来区别对待:

  1. 当一个类有接口的时候,就会选用 JDK 的动态代理。
  2. 当一个类没有实现接口的时候,就会选用 CGLIB 代理的方式。

以上就是Java两种方式实现动态代理的详细内容,更多关于Java 动态代理的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
  • 详解Java JDK动态代理
  • 详解Java Cglib动态代理
  • Java简单实现动态代理模式过程解析
  • Java JDK动态代理实现原理实例解析
  • 详细分析java 动态代理
  • Java使用JDK与Cglib动态代理技术统一管理日志记录
  • Java动态代理语法Proxy类原理详解
  • Java代理模式实例详解【静态代理与动态代理】
  • JAVA使用动态代理对象进行敏感字过滤代码实例
  • 详解JAVA动态代理
  • Java动态代理和反射机制详解
  • Java动态代理的两种实现方式详解【附相关jar文件下载】

相关文章

  • Java两种方式实现动态代理

    Java两种方式实现动态代理

    一、JDK动态代理 Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口、目标接口的类加载器以及 Invocation
    2020-10-10
  • Python判断变量是否是None写法代码实例

    Python判断变量是否是None写法代码实例

    代码中经常会有变量是否为None的判断,有三种主要的写法: 第一种是`if x is None`; 第二种是 `if not x:`; 第三种是`if not x is None`(这句这样理解更
    2020-10-10
  • java 枚举enum的用法(与在switch中的用法)

    java 枚举enum的用法(与在switch中的用法)

    实际开发中,很多人可能很少用枚举类型。更多的可能使用常量的方式代替。但枚举比起常量来说,含义更清晰,更容易理解,结构上也更加紧密。看其他人的博文都很详细,
    2020-10-10
  • Python批量获取并保存手机号归属地和运营商的示例

    Python批量获取并保存手机号归属地和运营商的示例

    从Excel读取一组手机号码,批量查询该手机号码的运营商和归属地,并将其追加到该记录的末尾。 import requests import json import xlrd from xlutils.copy impor
    2020-10-10
  • Python特殊属性property原理及使用方法解析

    Python特殊属性property原理及使用方法解析

    1 什么是特性property property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值 import math class Circle: def __init__(self,radius): #圆的半径
    2020-10-10
  • Python类绑定方法及非绑定方法实例解析

    Python类绑定方法及非绑定方法实例解析

    一、绑定方法   1.对象的绑定方法   首先我们明确一个知识点,凡是类中的方法或函数,默认情况下都是绑定给对象使用的。下面,我们通过实例,来慢慢解析绑
    2020-10-10
  • 在.NET中使用DiagnosticSource的方法

    在.NET中使用DiagnosticSource的方法

    前言 DiagnosticSource是一个非常有意思的且非常有用的API,对于这些API它们允许不同的库发送命名事件,并且它们也允许应用程序订阅这些事件并处理它们,它使我们
    2020-10-07
  • ASP.NET Core 实现基本认证的示例代码

    ASP.NET Core 实现基本认证的示例代码

    HTTP基本认证 在HTTP中,HTTP基本认证(Basic Authentication)是一种允许网页浏览器或其他客户端程序以(用户名:口令) 请求资源的身份验证方式,不要求cookie,sess
    2020-10-07
  • 9个JavaScript日常开发小技巧

    9个JavaScript日常开发小技巧

    1.生成指定范围的数字 在某些情况下,我们会创建一个处在两个数之间的数组。假设我们要判断某人的生日是否在某个范围的年份内,那么下面是实现它的一个很简单的方法
    2020-10-07
  • 浅谈 FTP、FTPS 与 SFTP的区别

    浅谈 FTP、FTPS 与 SFTP的区别

    无论是网盘还是云存储,上传都是一项很简单的操作。那些便捷好用的上传整理工具所用的 FTP 协议到底是什么意义,繁杂的模式又有何区别? 二狗子最近搭建了一个图片分
    2020-10-07

最新评论