• Java-基础-JDK动态代理


    1. 简介

    代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。

    比如:我们在调用底层框架方法时候,需要在调用方法的前后打印日志,或者做一些逻辑判断。此时我们无法去修改底层框架方法,那我们可以通过封装一个代理类,在代理类中实现对方法的处理,然后所有的客户端通过代理类去调用目标方法。

    其中这里有几个对象:

    1. 抽象角色:通过接口或者抽象类声明真实角色实现的业务方法,尽可能的保证代理对象的内部结构和目标对象一致,这样我们对代理对象的操作最终都可以转移到目标对象上。
    2. 代理角色/代理对象:实现抽象角色,是真实角色的代理,实现对目标方法的增强。
    3. 真实角色/目标对象:实现抽象角色,定义真实角色所要实现的业务逻辑,供代理角色调用。
    4. 调用角色/客户端:调用代理对象。

    2. 静态代理

    2.1 业务场景

    假设现在有这么个场景:王淑芬打算相亲,但是自己嘴笨,于是找到媒婆,希望媒婆帮自己找个帅哥,于是找到了张铁牛

    角色分析:

    • 王淑芬:目标对象(被代理的人)。
    • 媒婆:代理对象(代理了王淑芬,实现对目标方法的增强)。
    • 张铁牛:客户端(访问代理对象,即找媒婆)。
    • 抽象角色(相亲):媒婆王淑芬的共同目标-相亲成功。

    2.2 代码实现

    1. 相亲接口

      1. /**
      2. * 相亲抽象类
      3. *
      4. * @author ludangxin
      5. * @date 2021/9/25
      6. */
      7. public interface BlindDateService {
      8. /**
      9. * 聊天
      10. */
      11. void chat();
      12. }
    2. 目标对象

      1. /**
      2. * 王淑芬 - 目标对象
      3. *
      4. * @author ludangxin
      5. * @date 2021/9/25
      6. */
      7. @Slf4j
      8. public class WangShuFen implements BlindDateService {
      9. @Override
      10. public void chat() {
      11. log.info("美女:王淑芬~");
      12. }
      13. }

    3. 代理对象

      1. /**
      2. * 媒婆 - 代理对象
      3. *
      4. * @author ludangxin
      5. * @date 2021/9/25
      6. */
      7. @Slf4j
      8. public class WomanMatchmaker implements BlindDateService {
      9. private BlindDateService bs;
      10. public WomanMatchmaker() {
      11. this.bs = new WangShuFen();
      12. }
      13. @Override
      14. public void chat() {
      15. this.introduce();
      16. bs.chat();
      17. this.praise();
      18. }
      19. /**
      20. * 介绍
      21. */
      22. private void introduce() {
      23. log.info("媒婆:她的工作是web前端~");
      24. }
      25. /**
      26. * 夸人
      27. */
      28. private void praise() {
      29. log.info("媒婆:她就是有点害羞~");
      30. log.info("媒婆:你看她人长的漂亮,而且温柔贤惠,上的厅堂下的厨房~");
      31. log.info("媒婆:而且写bug超厉害~");
      32. }
      33. }

    4. 客户端

      1. /**
      2. * 张铁牛 - client
      3. *
      4. * @author ludangxin
      5. * @date 2021/9/25
      6. */
      7. public class ZhangTieNiu {
      8. public static void main(String[] args) {
      9. WomanMatchmaker wm = new WomanMatchmaker();
      10. wm.chat();
      11. }
      12. }
    5. 执行方法输出内容如下:

      1. 22:44:51.184 [main] INFO proxy.staticp.WomanMatchmaker - 媒婆:她的工作是web前端~
      2. 22:44:51.191 [main] INFO proxy.staticp.WangShuFen - 美女:你好,我叫王淑芬~
      3. 22:44:51.191 [main] INFO proxy.staticp.WomanMatchmaker - 媒婆:她就是有点害羞~
      4. 22:44:51.191 [main] INFO proxy.staticp.WomanMatchmaker - 媒婆:你看她人长的漂亮,而且温柔贤惠,上的厅堂下的厨房~
      5. 22:44:51.191 [main] INFO proxy.staticp.WomanMatchmaker - 媒婆:而且写bug超厉害~

    2.3 小节

    好处:

    1. 耦合性降低。因为加入了代理类,调用者只用关心代理类即可,降低了调用者与目标类的耦合度。
    2. 指责清晰,目标对象只关心真实的业务逻辑。代理对象只负责对目标对象的增强。调用者只关心代理对象的执行结果。
    3. 代理对象实现了对目标方法的增强。也就是说:代理对象 = 增强代码 + 目标对象

    缺陷:

    每一个目标类都需要写对应的代理类。如果当前系统已经有成百上千个类,工作量大太。所以,能不能不用写那么多代理类,就能实现对目标方法的增强呢?

    3. 动态代理

    我们常见的动态代理一般有两种:JDK动态代理CGLib动态代理,本章只讲JDK动态代理

    在了解JDK动态代理之前,先了解两个重要的类。

    3.1 Proxy

    从JDK的帮助文档中可知:

    Proxy提供了创建动态代理和实例的静态方法。即:

    Proxy::newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)

    参数:

    • loader - 定义代理类的类加载器

    • interfaces - 代理类要实现的接口列表

    • h - 指派方法调用的调用处理程序

    前两个参数没啥好说的,主要还需要了解下InvocationHandler

    3.2 InvocationHandler

    从JDK的帮助文档中可知:

    InvocationHandler是一个接口,并且是调用处理逻辑实现的接口。在调用代理对象方法时,实际上是调用实现接口的invoke方法。

    Object InvocationHandler::invoke(Object proxy, Method method, Object[] args)

    参数:

    • proxy - 调用方法的代理实例。

    • method - 对应于在代理实例上调用的接口方法的 Method 实例。 Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。

    • args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integer 或 java.lang.Boolean)的实例中。

    返回值:

    从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 ClassCastException

    3.3 代码实现

    1. 创建动态代理类

      1. import lombok.extern.slf4j.Slf4j;
      2. import java.lang.reflect.InvocationHandler;
      3. import java.lang.reflect.Method;
      4. import java.lang.reflect.Proxy;
      5. import java.util.Arrays;
      6. /**
      7. * 媒婆 - 代理对象
      8. *
      9. * @author ludangxin
      10. * @date 2021/9/25
      11. */
      12. @Slf4j
      13. public class ProxyTest {
      14. public static Object getProxy(final Object target) {
      15. return Proxy.newProxyInstance(
      16. //类加载器
      17. target.getClass().getClassLoader(),
      18. //让代理对象和目标对象实现相同接口
      19. target.getClass().getInterfaces(),
      20. //代理对象的方法最终都会被JVM导向它的invoke方法
      21. new InvocationHandler() {
      22. @Override
      23. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
      24. log.info("proxy ===》{}", proxy.getClass().toGenericString());
      25. log.info("method ===》{}", method.toGenericString());
      26. log.info("args ===》{}", Arrays.toString(args));
      27. introduce();
      28. // 调用目标方法
      29. Object result = method.invoke(target, args);
      30. praise();
      31. return result;
      32. }
      33. });
      34. }
      35. /**
      36. * 介绍
      37. */
      38. private static void introduce() {
      39. log.info("媒婆:她的工作是web前端~");
      40. }
      41. /**
      42. * 夸人
      43. */
      44. private static void praise() {
      45. log.info("媒婆:她就是有点害羞~");
      46. log.info("媒婆:你看她人长的漂亮,而且温柔贤惠,上的厅堂下的厨房~");
      47. log.info("媒婆:而且写bug超厉害~");
      48. }
      49. }
    2. 客户端

      1. /**
      2. * 张铁牛 - client
      3. *
      4. * @author ludangxin
      5. * @date 2021/9/25
      6. */
      7. public class ZhangTieNiu {
      8. public static void main(String[] args) {
      9. BlindDateService proxy = (BlindDateService)ProxyTest.getProxy(new WangShuFen());
      10. proxy.chat();
      11. }
      12. }
    3. 执行方法输出内容如下:

      1. 21:29:22.222 [main] INFO proxy.dynamic.ProxyTest - proxy ===》public final class com.sun.proxy.$Proxy0
      2. 21:29:22.229 [main] INFO proxy.dynamic.ProxyTest - method ===》public abstract void proxy.dynamic.BlindDateService.chat()
      3. 21:29:22.229 [main] INFO proxy.dynamic.ProxyTest - args ===》null
      4. 21:29:22.229 [main] INFO proxy.dynamic.ProxyTest - 媒婆:她的工作是web前端~
      5. 21:29:22.229 [main] INFO proxy.dynamic.WangShuFen - 美女:你好,我叫王淑芬~
      6. 21:29:22.230 [main] INFO proxy.dynamic.ProxyTest - 媒婆:她就是有点害羞~
      7. 21:29:22.230 [main] INFO proxy.dynamic.ProxyTest - 媒婆:你看她人长的漂亮,而且温柔贤惠,上的厅堂下的厨房~
      8. 21:29:22.230 [main] INFO proxy.dynamic.ProxyTest - 媒婆:而且写bug超厉害~

    3.4 小节

    其实代码很简单,只需要通过jdk提供的代理类创建一个代理类,再通过代理类去调用目标方法,并实现对方法的增强即可。

    从打印的日志中可以看出JDK生成真实的代理对象为com.sun.proxy.$Proxy0,那我们能不能查看下生成的代理对象的源码呢?答案肯定是可以的。我们可以借助JDK默认提供的ProxyGenerator::generateProxyClass()来输出动态生成的代理对象。

    改造下动态代理类如下:

    添加输出动态代理类的方法。

    1. package proxy.dynamic;
    2. import lombok.extern.slf4j.Slf4j;
    3. import sun.misc.ProxyGenerator;
    4. import java.io.File;
    5. import java.io.FileOutputStream;
    6. import java.lang.reflect.InvocationHandler;
    7. import java.lang.reflect.Method;
    8. import java.lang.reflect.Proxy;
    9. import java.util.Arrays;
    10. /**
    11. * 媒婆 - 代理对象
    12. *
    13. * @author ludangxin
    14. * @date 2021/9/25
    15. */
    16. @Slf4j
    17. public class ProxyTest {
    18. public static Object getProxy(final Object target) {
    19. return Proxy.newProxyInstance(
    20. //类加载器
    21. target.getClass().getClassLoader(),
    22. //让代理对象和目标对象实现相同接口
    23. target.getClass().getInterfaces(),
    24. //代理对象的方法最终都会被JVM导向它的invoke方法
    25. new InvocationHandler() {
    26. @Override
    27. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    28. generateProxyClass(proxy);
    29. log.info("proxy ===》{}", proxy.getClass().toGenericString());
    30. log.info("method ===》{}", method.toGenericString());
    31. log.info("args ===》{}", Arrays.toString(args));
    32. introduce();
    33. Object result = method.invoke(target, args);
    34. praise();
    35. return result;
    36. }
    37. });
    38. }
    39. /**
    40. * 输出动态生成的代理类
    41. * @param proxy 代理实例
    42. */
    43. private static void generateProxyClass(Object proxy) {
    44. byte[] bytes = ProxyGenerator.generateProxyClass(proxy.getClass().getName(), proxy.getClass().getInterfaces());
    45. File file = new File("/Users/ludangxin/workspace/idea/test/target/classes/proxy/dynamic/proxy.class");
    46. try {
    47. FileOutputStream fileOutputStream = new FileOutputStream(file);
    48. fileOutputStream.write(bytes);
    49. } catch(Exception e) {
    50. e.printStackTrace();
    51. }
    52. }
    53. ...
    54. }

    再次执行调用方法,输出类如下:

    类信息如下:

    看完代理类的方法是不是恍然大悟,我们再整理下调用过程:

    动态代理:动态的生成代理对象。

    4. 总结

    其实无论是静态代理还是动态代理本质都是最终生成代理对象,区别在于静态代理对象需要人手动生成,而动态代理对象是运行时,JDK通过反射动态生成的代理类最终构造的对象,JDK生成的类在加载到内存之后就删除了,所以看不到类文件。

  • 相关阅读:
    一文速通Sentinel熔断及降级规则
    java计算机毕业设计废旧物品回收管理系统MyBatis+系统+LW文档+源码+调试部署
    Conda 环境移植 (两种方式)
    js如何实现一个简单的节流函数?
    SQL语句中 LEFT JOIN 后 ON 和 WHERE 的区别
    【Linux】进程等待wait/waitpid && status详解 && (非)阻塞等待(代码)
    Spark Sql之count(distinct)分析&&学习&&验证
    Java项目:台球室计费管理系统(java+SSM+JSP+HTML+JavaScript+mysql)
    pycharm安装opencv-python报错
    selenium 知网爬虫之根据【关键词】获取文献信息
  • 原文地址:https://blog.csdn.net/guanshengg/article/details/126436556