欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 文旅 > 八卦 > 什么是动态代理?动态代理和静态代理的区别

什么是动态代理?动态代理和静态代理的区别

2025/7/4 1:29:36 来源:https://blog.csdn.net/weixin_73355042/article/details/146885653  浏览:    关键词:什么是动态代理?动态代理和静态代理的区别

在 Java 中,动态代理是一种通过反射机制在运行时动态生成代理对象的技术,它能够在不修改原有代码的基础上增强目标对象的功能。以下是关于动态代理的详细说明:


一、静态代理 vs 动态代理

1. 静态代理
  • 定义:手动编写代理类,代理类需要实现与目标类相同的接口,并在方法中调用目标对象的方法。

  • 缺点

    • 每个目标类需要一个对应的代理类,代码冗余。
    • 接口或目标类变更时,代理类需要同步修改。
  • 示例

    interface UserService {void save();
    }class UserServiceImpl implements UserService {public void save() { System.out.println("保存用户"); }
    }// 静态代理类
    class UserServiceProxy implements UserService {private UserService target;public UserServiceProxy(UserService target) { this.target = target; }public void save() {System.out.println("前置操作");target.save();System.out.println("后置操作");}
    }
    
2. 动态代理
  • 定义:在运行时动态生成代理类,无需手动编写代理类。
  • 优点
    • 一个代理类可以代理多个目标类。
    • 代码复用性高,维护成本低。
  • 核心机制:通过反射(java.lang.reflect.Proxy)或字节码增强(如 CGLIB)生成代理对象。

二、Java 动态代理的实现方式

1. JDK 动态代理(基于接口)
  • 要求:目标对象必须实现至少一个接口。

  • 核心类

    • java.lang.reflect.Proxy:生成代理对象。
    • java.lang.reflect.InvocationHandler:定义代理逻辑。
  • 实现步骤

    1. 定义接口和实现类(目标对象)。
    2. 实现 InvocationHandler 接口,编写增强逻辑。
    3. 通过 Proxy.newProxyInstance() 生成代理对象。
  • 代码示例

    interface UserService { void save(); }class UserServiceImpl implements UserService {public void save() { System.out.println("保存用户"); }
    }class MyInvocationHandler implements InvocationHandler {private Object target;public MyInvocationHandler(Object target) { this.target = target; }public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("前置操作");Object result = method.invoke(target, args); // 反射调用目标方法System.out.println("后置操作");return result;}
    }// 生成代理对象
    UserService proxy = (UserService) Proxy.newProxyInstance(UserService.class.getClassLoader(),new Class[]{UserService.class},new MyInvocationHandler(new UserServiceImpl())
    );
    proxy.save(); // 调用代理方法
    
2. CGLIB 动态代理(基于继承)
  • 要求:目标类不需要实现接口,通过继承目标类生成子类代理。

  • 依赖:需要引入 cglib 库(Spring Core 已内置)。

  • 核心类

    • net.sf.cglib.proxy.Enhancer:生成代理对象。
    • net.sf.cglib.proxy.MethodInterceptor:定义代理逻辑。
  • 代码示例

    java

    复制

    class UserService {public void save() { System.out.println("保存用户"); }
    }class MyMethodInterceptor implements MethodInterceptor {public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("前置操作");Object result = proxy.invokeSuper(obj, args); // 调用父类方法System.out.println("后置操作");return result;}
    }// 生成代理对象
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(UserService.class);
    enhancer.setCallback(new MyMethodInterceptor());
    UserService proxy = (UserService) enhancer.create();
    proxy.save();
    

三、动态代理的应用场景

  1. AOP 编程:如 Spring AOP 的切面(日志、事务、权限控制)。
  2. RPC 框架:远程方法调用的客户端代理。
  3. 延迟加载:Hibernate 的延迟加载实现。
  4. 单元测试:Mock 对象的动态生成。

四、JDK 动态代理 vs CGLIB 动态代理

特性JDK 动态代理CGLIB 动态代理
依赖目标类必须实现接口无接口要求
生成方式反射生成代理类继承目标类生成子类
性能调用较快,生成较慢生成较快,调用较慢
局限性无法代理未实现接口的类无法代理 final 类/方法
应用框架Spring AOP(默认接口场景)Spring AOP(非接口场景)

五、总结

  • 静态代理适合简单场景,但代码冗余。
  • 动态代理通过反射或字节码技术动态生成代理类,灵活且高效。
  • JDK 动态代理基于接口,CGLIB基于继承,根据需求选择合适方式。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词