Java静态代理和动态代理在实现上有何差异

静态代理在编译期手动编写代理类,代理类与目标类实现同一接口,扩展性差;动态代理在运行时通过反射机制自动生成代理对象,JDK动态代理基于接口,CGLIB通过继承实现,灵活性高,适用于统一处理横切逻辑,是Spring AOP的基础。

静态代理和动态代理的核心区别在于代理类的生成时机和实现方式。静态代理在编译期就已经确定,而动态代理是在运行时动态创建的。

静态代理:编译期确定,手动编写代理类

静态代理要求程序员手动创建一个代理类,这个类实现与目标对象相同的接口,并持有目标对象的引用。代理类在程序运行前就已经存在。

特点:

  • 代理类和被代理类在编译时就已确定
  • 每个被代理类都需要一个对应的代理类
  • 代码重复较多,扩展性差
  • 适用于代理逻辑固定、被代理类较少的场景
例如:UserService 接口有实现类UserServiceImpl,静态代理需要写一个 UserServiceProxy 类,也实现 UserService,并在方法中调用目标方法前后加入逻辑。

动态代理:运行时生成,无需提前编写代理类

动态代理在程序运行过程中,根据需要动态生成代理对象。Java 提供了两种常见实现方式:JDK 动态代理和 CGLIB。

JDK 动态代理基于接口,使用 java.lang.reflect.Proxy 类和 InvocationHandler 接口;CGLIB 则通过继承方式实现代理,适用于没有接口的类。

特点:

  • 代理类在运行时自动生成,无需手动编写
  • 一个 InvocationHandler 可以处理多个接口的代理逻辑
  • 灵活性高,易于统一管理横切逻辑(如日志、事务)
  • JDK 动态代理只能代理实现了接口的类
例如:通过 Proxy.newProxyInstance() 创建代理对象,在 invoke() 方法中定义前置、后置操作,目标方法执行前后均可增强。

关键差异对比

从实现角度看:

  • 静态代理依赖手工编码,动态代理依赖反射机制
  • 静态代理增删功能需修改源码,动态代理只需调整拦截逻辑
  • 动态代理更符合开闭原则,对扩展开放,对修改封闭
基本上就这些。静态代理简单直接但不够灵活,动态代理通用性强,是 Spring AOP 等框架的基础。选择哪种方式取决于具体需求和设计目标。