【动态代理】—— JDK和cglib的动态代理

x33g5p2x  于2022-03-20 转载在 其他  
字(2.2k)|赞(0)|评价(0)|浏览(711)

概述

设计模式中有一个代理模式,它为其他对象提供一种代理以控制对这个对象的访问。
关于代理模式可以参考:【每天一个java设计模式(七)】 - 代理模式:https://blog.csdn.net/weixin_43598687/article/details/122072282
代理模式是指的静态代理。使用静态代理很容易就完成了对一个类的代理操作。但是静态代理的缺点也暴露了出来:由于代理只能为一个类服务,如果需要代理的类很多,那么就需要编写大量的代理类,比较繁琐。

动态代理可以在程序运行期间,在不修改源码的情况下对方法进行功能增强。

下来给出两种方式的动态代理的实现:JDK、cglib

JDK的动态代理

JDK提供了java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy类,基于接口和反射技术可以实现动态代理。【java中的反射机制解析:https://blog.csdn.net/weixin_43598687/article/details/121890395】

下面是JDK的动态代理的简单实现:

1. 创建一个目标类和接口

  1. public interface TargetInterface {
  2. public void coreWork();
  3. }
  1. public class Target implements TargetInterface {
  2. @Override
  3. public void coreWork() {
  4. System.out.println("===核心业务方法运行===");
  5. }
  6. }

2. 创建一个增强方法类

  1. public class Advice {
  2. public void before(){
  3. System.out.println("对核心业务方法执行前的增强......");
  4. }
  5. public void after(){
  6. System.out.println("后置增强......");
  7. }
  8. }

3. 动态代理测试

  1. public class ProxyMain {
  2. public static void main(String[] args) {
  3. // 目标对象
  4. Target target = new Target();
  5. // 增强对象
  6. Advice advice = new Advice();
  7. TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
  8. //目标对象类加载器
  9. target.getClass().getClassLoader(),
  10. // 目标对象相同的接口字节码对象数组
  11. target.getClass().getInterfaces(),
  12. new InvocationHandler() {
  13. @Override
  14. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  15. advice.before(); //前置增强
  16. Object invoke = method.invoke(target, args);
  17. advice.after(); //后置增强
  18. return invoke;
  19. }
  20. }
  21. );
  22. // 通过代理类执行目标类的方法
  23. proxy.coreWork();
  24. }
  25. }

通过代理对象执行目标对象的方法,同时也执行了增强方法。

cglib的动态代理

基于父类的动态代理技术

1. 创建一个目标类

  1. public class Target{
  2. public void coreWork() {
  3. System.out.println("===核心业务方法运行===");
  4. }
  5. }

2. 创建一个增强方法类

  1. public class Advice {
  2. public void before(){
  3. System.out.println("对核心业务方法执行前的增强......");
  4. }
  5. public void after(){
  6. System.out.println("后置增强......");
  7. }
  8. }

3. 动态代理测试

  1. public class ProxyMain {
  2. public static void main(String[] args) {
  3. Target target = new Target();
  4. Advice advice = new Advice();
  5. // 1. 创建增强器
  6. Enhancer enhancer = new Enhancer();
  7. // 2. 设置增强目标类
  8. enhancer.setSuperclass(target.getClass());
  9. // 3. 设置回调
  10. enhancer.setCallback(new MethodInterceptor() {
  11. @Override
  12. public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
  13. advice.before(); // 前置增强
  14. Object invoke = method.invoke(target, args);
  15. advice.after(); //后置增强
  16. return invoke;
  17. }
  18. });
  19. // 4. 创建代理
  20. Target targetProxy = (Target) enhancer.create();
  21. targetProxy.coreWork();
  22. }
  23. }

相关文章