服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 面试官提问:什么是动态代理?

面试官提问:什么是动态代理?

2021-12-08 22:29Java极客技术鸭血粉丝Tang Java教程

据史料记载,代理这个词最早出现在代理商这个行业,所谓代理商,简而言之,其实就是帮助企业或者老板打理生意,自己本身不做生产任何商品。

面试官提问:什么是动态代理?

一、介绍

何谓代理?

据史料记载,代理这个词最早出现在代理商这个行业,所谓代理商,简而言之,其实就是帮助企业或者老板打理生意,自己本身不做生产任何商品。

举个例子,我们去火车站买票的时候,人少老板一个人还忙的过来,但是人一多的话,就会非常拥挤,于是就有了各种代售点,我们可以从代售点买车票,从而加快老板的卖票速度。

代售点的出现,可以说,很直观的帮助老板提升了用户购票体验。

站在软件设计的角度,其实效果也是一样的,采用代理模式的编程,能显著的增强原有的功能和简化方法调用方式。

面试官提问:什么是动态代理?

在介绍动态代理之前,我们先来聊解静态代理。

二、静态代理

下面,我们以两数相加为例,实现过程如下!

接口类

  1. public interface Calculator {
  2. /**
  3. * 计算两个数之和
  4. * @param num1
  5. * @param num2
  6. * @return
  7. */
  8. Integer add(Integer num1, Integer num2);
  9. }

目标对象

  1. public class CalculatorImpl implements Calculator {
  2. @Override
  3. public Integer add(Integer num1, Integer num2) {
  4. Integer result = num1 + num2;
  5. return result;
  6. }
  7. }

代理对象

  1. public class CalculatorProxyImpl implements Calculator {
  2. private Calculator calculator;
  3. @Override
  4. public Integer add(Integer num1, Integer num2) {
  5. //方法调用前,可以添加其他功能....
  6. Integer result = calculator.add(num1, num2);
  7. //方法调用后,可以添加其他功能....
  8. return result;
  9. }
  10. public CalculatorProxyImpl(Calculator calculator) {
  11. this.calculator = calculator;
  12. }
  13. }

测试类

  1. public class CalculatorProxyClient {
  2. public static void main(String[] args) {
  3. //目标对象
  4. Calculator target = new CalculatorImpl();
  5. //代理对象
  6. Calculator proxy = new CalculatorProxyImpl(target);
  7. Integer result = proxy.add(1,2);
  8. System.out.println("相加结果:" + result);
  9. }
  10. }

输出结果

  1. 相加结果:3

通过这种代理方式,最大的优点就是:可以在不修改目标对象的前提下,扩展目标对象的功能。

但也有缺点:需要代理对象和目标对象实现一样的接口,因此,当目标对象扩展新的功能时,代理对象也要跟着一起扩展,不易维护!

三、动态代理

动态代理,其实本质也是为了解决上面当目标对象扩展新功能时,代理对象也需要跟着一起扩展的痛点问题而生。

那它是怎么解决的呢?

以 JDK 为例,当需要给某个目标对象添加代理处理的时候,JDK 会在内存中动态的构建代理对象,从而实现对目标对象的代理功能。

下面,我们还是以两数相加为例,介绍具体的玩法!

3.1、JDK 中生成代理对象的玩法

创建接口

  1. public interface JdkCalculator {
  2. /**
  3. * 计算两个数之和
  4. * @param num1
  5. * @param num2
  6. * @return
  7. */
  8. Integer add(Integer num1, Integer num2);
  9. }

目标对象

  1. public class JdkCalculatorImpl implements JdkCalculator {
  2. @Override
  3. public Integer add(Integer num1, Integer num2) {
  4. Integer result = num1 + num2;
  5. return result;
  6. }
  7. }

动态代理对象

  1. public class JdkProxyFactory {
  2. /**
  3. * 维护一个目标对象
  4. */
  5. private Object target;
  6. public JdkProxyFactory(Object target) {
  7. this.target = target;
  8. }
  9. public Object getProxyInstance(){
  10. Object proxyClassObj = Proxy.newProxyInstance(target.getClass().getClassLoader(),
  11. target.getClass().getInterfaces(),
  12. new InvocationHandler(){
  13. @Override
  14. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  15. System.out.println("方法调用前,可以添加其他功能....");
  16. // 执行目标对象方法
  17. Object returnValue = method.invoke(target, args);
  18. System.out.println("方法调用后,可以添加其他功能....");
  19. return returnValue;
  20. }
  21. });
  22. return proxyClassObj;
  23. }
  24. }

测试类

  1. public class TestJdkProxy {
  2. public static void main(String[] args) {
  3. //目标对象
  4. JdkCalculator target = new JdkCalculatorImpl();
  5. System.out.println(target.getClass());
  6. //代理对象
  7. JdkCalculator proxyClassObj = (JdkCalculator) new JdkProxyFactory(target).getProxyInstance();
  8. System.out.println(proxyClassObj.getClass());
  9. //执行代理方法
  10. Integer result = proxyClassObj.add(1,2);
  11. System.out.println("相加结果:" + result);
  12. }
  13. }

输出结果

  1. class com.example.java.proxy.jdk1.JdkCalculatorImpl
  2. class com.sun.proxy.$Proxy0
  3. 方法调用前,可以添加其他功能....
  4. 方法调用后,可以添加其他功能....
  5. 相加结果:3

采用 JDK 技术动态创建interface实例的步骤如下:

  1. 1. 首先定义一个 InvocationHandler 实例,它负责实现接口的方法调用
  2. 2. 通过 Proxy.newProxyInstance() 创建 interface 实例,它需要 3 个参数:
  3. (1)使用的 ClassLoader,通常就是接口类的 ClassLoader
  4. (2)需要实现的接口数组,至少需要传入一个接口进去;
  5. (3)用来处理接口方法调用的 InvocationHandler 实例。
  6. 3. 将返回的 Object 强制转型为接口

动态代理实际上是 JVM 在运行期动态创建class字节码并加载的过程,它并没有什么黑魔法技术,把上面的动态代理改写为静态实现类大概长这样:

  1. public class JdkCalculatorDynamicProxy implements JdkCalculator {
  2. private InvocationHandler handler;
  3. public JdkCalculatorDynamicProxy(InvocationHandler handler) {
  4. this.handler = handler;
  5. }
  6. public void add(Integer num1, Integer num2) {
  7. handler.invoke(
  8. this,
  9. JdkCalculator.class.getMethod("add", Integer.class, Integer.class),
  10. new Object[] { num1, num2 });
  11. }
  12. }

本质就是 JVM 帮我们自动编写了一个上述类(不需要源码,可以直接生成字节码)。

3.2、cglib 生成代理对象的玩法

除了 jdk 能实现动态的创建代理对象以外,还有一个非常有名的第三方框架:cglib,它也可以做到运行时在内存中动态生成一个子类对象从而实现对目标对象功能的扩展。

cglib 特点如下:

cglib 不仅可以代理接口还可以代理类,而 JDK 的动态代理只能代理接口

cglib 是一个强大的高性能的代码生成包,它广泛的被许多 AOP 的框架使用,例如我们所熟知的 Spring AOP,cglib 为他们提供方法的 interception(拦截)。

CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类,速度非常快。

在使用 cglib 之前,我们需要添加依赖包,如果你已经有spring-core的jar包,则无需引入,因为spring中包含了cglib。

  1. cglib
  2. cglib
  3. 3.2.5

下面,我们还是以两数相加为例,介绍具体的玩法!

  1. public interface CglibCalculator {
  2. /**
  3. * 计算两个数之和
  4. * @param num1
  5. * @param num2
  6. * @return
  7. */
  8. Integer add(Integer num1, Integer num2);
  9. }

目标对象

  1. public class CglibCalculatorImpl implements CglibCalculator {
  2. @Override
  3. public Integer add(Integer num1, Integer num2) {
  4. Integer result = num1 + num2;
  5. return result;
  6. }
  7. }

动态代理对象

  1. public class CglibProxyFactory implements MethodInterceptor {
  2. /**
  3. * 维护一个目标对象
  4. */
  5. private Object target;
  6. public CglibProxyFactory(Object target) {
  7. this.target = target;
  8. }
  9. /**
  10. * 为目标对象生成代理对象
  11. * @return
  12. */
  13. public Object getProxyInstance() {
  14. //工具类
  15. Enhancer en = new Enhancer();
  16. //设置父类
  17. en.setSuperclass(target.getClass());
  18. //设置回调函数
  19. en.setCallback(this);
  20. //创建子类对象代理
  21. return en.create();
  22. }
  23. @Override
  24. public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  25. System.out.println("方法调用前,可以添加其他功能....");
  26. // 执行目标对象方法
  27. Object returnValue = method.invoke(target, args);
  28. System.out.println("方法调用后,可以添加其他功能....");
  29. return returnValue;
  30. }
  31. }

测试类

  1. public class TestCglibProxy {
  2. public static void main(String[] args) {
  3. //目标对象
  4. CglibCalculator target = new CglibCalculatorImpl();
  5. System.out.println(target.getClass());
  6. //代理对象
  7. CglibCalculator proxyClassObj = (CglibCalculator) new CglibProxyFactory(target).getProxyInstance();
  8. System.out.println(proxyClassObj.getClass());
  9. //执行代理方法
  10. Integer result = proxyClassObj.add(1,2);
  11. System.out.println("相加结果:" + result);
  12. }
  13. }

输出结果

  1. class com.example.java.proxy.cglib1.CglibCalculatorImpl
  2. class com.example.java.proxy.cglib1.CglibCalculatorImpl$$EnhancerByCGLIB$$3ceadfe4
  3. 方法调用前,可以添加其他功能....
  4. 方法调用后,可以添加其他功能....
  5. 相加结果:3

将 cglib 生成的代理类改写为静态实现类大概长这样:

  1. public class CglibCalculatorImplByCGLIB extends CglibCalculatorImpl implements Factory {
  2. private static final MethodInterceptor methodInterceptor;
  3. private static final Method method;
  4. public final Integer add(Integer var1, Integer var2) {
  5. return methodInterceptor.intercept(this, method, new Object[]{var1, var2}, methodProxy);
  6. }
  7. //....
  8. }

其中,拦截思路与 JDK 类似,都是通过一个接口方法进行拦截处理!

在上文中咱们还介绍到了,cglib 不仅可以代理接口还可以代理类,下面我们试试代理类。

  1. public class CglibCalculatorClass {
  2. /**
  3. * 计算两个数之和
  4. * @param num1
  5. * @param num2
  6. * @return
  7. */
  8. public Integer add(Integer num1, Integer num2) {
  9. Integer result = num1 + num2;
  10. return result;
  11. }
  12. }

测试类

  1. public class TestCglibProxyClass {
  2. public static void main(String[] args) {
  3. //目标对象
  4. CglibCalculatorClass target = new CglibCalculatorClass();
  5. System.out.println(target.getClass());
  6. //代理对象
  7. CglibCalculatorClass proxyClassObj = (CglibCalculatorClass) new CglibProxyFactory(target).getProxyInstance();
  8. System.out.println(proxyClassObj.getClass());
  9. //执行代理方法
  10. Integer result = proxyClassObj.add(1,2);
  11. System.out.println("相加结果:" + result);
  12. }
  13. }

输出结果

  1. class com.example.java.proxy.cglib1.CglibCalculatorClass
  2. class com.example.java.proxy.cglib1.CglibCalculatorClass$$EnhancerByCGLIB$$e68ff36c
  3. 方法调用前,可以添加其他功能....
  4. 方法调用后,可以添加其他功能....
  5. 相加结果:3

四、静态织入

在上文中,我们介绍的代理方案都是在代码运行时动态的生成class文件达到动态代理的目的。

回到问题的本质,其实动态代理的技术目的,主要为了解决静态代理模式中当目标接口发生了扩展,代理类也要跟着一遍变动的问题,避免造成了工作伤的繁琐和复杂。

在 Java 生态里面,还有一个非常有名的第三方代理框架,那就是AspectJ,AspectJ通过特定的编译器可以将目标类编译成class字节码的时候,在方法周围加上业务逻辑,从而达到静态代理的效果。

采用AspectJ进行方法植入,主要有四种:

  • 方法调用前拦截
  • 方法调用后拦截
  • 调用方法结束拦截
  • 抛出异常拦截

使用起来也非常简单,首先是在项目中添加AspectJ编译器插件。

  1. org.codehaus.mojo
  2. aspectj-maven-plugin
  3. 1.5
  4. compile
  5. test-compile
  6. 1.6
  7. 1.6
  8. UTF-8
  9. 1.6
  10. true
  11. true

然后,编写一个方法,准备进行代理。

  1. @RequestMapping({"/hello"})
  2. public String hello(String name) {
  3. String result = "Hello World";
  4. System.out.println(result);
  5. return result;
  6. }

编写代理配置类

  1. @Aspect
  2. public class ControllerAspect {
  3. /***
  4. * 定义切入点
  5. */
  6. @Pointcut("execution(* com.example.demo.web..*.*(..))")
  7. public void methodAspect(){}
  8. /**
  9. * 方法调用前拦截
  10. */
  11. @Before("methodAspect()")
  12. public void before(){
  13. System.out.println("代理 -> 调用方法执行之前......");
  14. }
  15. /**
  16. * 方法调用后拦截
  17. */
  18. @After("methodAspect()")
  19. public void after(){
  20. System.out.println("代理 -> 调用方法执行之后......");
  21. }
  22. /**
  23. * 调用方法结束拦截
  24. */
  25. @AfterReturning("methodAspect()")
  26. public void afterReturning(){
  27. System.out.println("代理 -> 调用方法结束之后......");
  28. }
  29. /**
  30. * 抛出异常拦截
  31. */
  32. @AfterThrowing("methodAspect()")
  33. public void afterThrowing() {
  34. System.out.println("代理 -> 调用方法异常......");
  35. }
  36. }

编译后,hello方法会变成这样。

  1. @RequestMapping({"/hello"})
  2. public String hello(Integer name) throws SQLException {
  3. JoinPoint var2 = Factory.makeJP(ajc$tjp_0, this, this, name);
  4. Object var7;
  5. try {
  6. Object var5;
  7. try {
  8. //调用before
  9. Aspectj.aspectOf().doBeforeTask2(var2);
  10. String result = "Hello World";
  11. System.out.println(result);
  12. var5 = result;
  13. } catch (Throwable var8) {
  14. Aspectj.aspectOf().after(var2);
  15. throw var8;
  16. }
  17. //调用after
  18. Aspectj.aspectOf().after(var2);
  19. var7 = var5;
  20. } catch (Throwable var9) {
  21. //调用抛出异常
  22. Aspectj.aspectOf().afterthrowing(var2);
  23. throw var9;
  24. }
  25. //调用return
  26. Aspectj.aspectOf().afterRutuen(var2);
  27. return (String)var7;
  28. }

很显然,代码被AspectJ编译器修改了,AspectJ并不是动态的在运行时生成代理类,而是在编译的时候就植入代码到class文件。

由于是静态织入的,所以性能相对来说比较好!

五、小结

看到上面的介绍静态织入方案,跟我们现在使用Spring AOP的方法极其相似,可能有的同学会发出疑问,我们现在使用的Spring AOP动态代理,到底是动态生成的还是静态织入的呢?

实际上,Spring AOP代理是对JDK代理和CGLIB代理做了一层封装,同时引入了AspectJ中的一些注解@pointCut、@after,@before等等,本质是使用的动态代理技术。

总结起来就三点:

如果目标是接口的话,默认使用 JDK 的动态代理技术;

如果目标是类的话,使用 cglib 的动态代理技术;

引入了AspectJ中的一些注解@pointCut、@after,@before,主要是为了简化使用,跟AspectJ的关系并不大;

那为什么Spring AOP不使用AspectJ这种静态织入方案呢?

虽然AspectJ编译器非常强,性能非常高,但是只要目标类发生了修改就需要重新编译,主要原因可能还是AspectJ的编译器太过于复杂,还不如动态代理来的省心!

六、参考

1、Java三种代理模式:静态代理、动态代理和cglib代理

2、Java 动态代理作用是什么?

原文链接:https://mp.weixin.qq.com/s/bCGncYhu41OJXKCzGeS5Nw

延伸 · 阅读

精彩推荐
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7482021-02-04