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

在 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基于继承,根据需求选择合适方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值