JAVA 动态代理

一句话概括:不写死代理类,在程序运行时自动生成代理对象,用来在不修改原代码的前提下,统一加功能(日志、事务、权限、统计等)。


一、两种动态代理

  1. JDK 动态代理

    • 基于 接口
    • 被代理类 必须实现接口
    • Java 原生,无需依赖
  2. CGLIB 动态代理

    • 基于 继承
    • 可以代理 没有接口的类
    • 需要引入 cglib 依赖(Spring 内置)

二、JDK 动态代理(最常用)

核心类

  • InvocationHandler:处理逻辑的接口
  • Proxy.newProxyInstance():生成代理对象

步骤

  1. 定义 接口
  2. 实现类(目标对象)
  3. InvocationHandler 实现类 ,在 invoke 里加增强逻辑
  4. 通过 Proxy 生成代理对象

极简示例

1. 接口

java

运行

复制代码
public interface UserService {
    void add();
}
2. 实现类

java

运行

复制代码
public class UserServiceImpl implements UserService {
    @Override
    public void add() {
        System.out.println("添加用户");
    }
}
3. 代理处理器

java

运行

复制代码
public class MyHandler implements InvocationHandler {
    private Object target;

    public MyHandler(Object target) {
        this.target = target;
    }

    @Override
    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;
    }
}
4. 测试

java

运行

复制代码
public class Test {
    public static void main(String[] args) {
        UserService target = new UserServiceImpl();

        UserService proxy = (UserService) Proxy.newProxyInstance(
            target.getClass().getClassLoader(),
            target.getClass().getInterfaces(),
            new MyHandler(target)
        );

        proxy.add();
    }
}

输出

plaintext

复制代码
方法前:日志
添加用户
方法后:提交事务

三、CGLIB 动态代理(无接口也能用)

核心类

  • MethodInterceptor
  • Enhancer

示例(关键代码)

java

运行

复制代码
public class CglibInterceptor implements MethodInterceptor {
    @Override
    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 CglibInterceptor());
UserService proxy = (UserService) enhancer.create();
proxy.add();

四、常见面试要点

  1. JDK vs CGLIB

    • JDK:接口、快、原生
    • CGLIB:继承、可代理普通类、Spring 常用
  2. 动态代理用途

    • Spring AOP
    • MyBatis Mapper 代理
    • 全局日志、权限校验、事务管理
  3. 为什么 JDK 必须要接口? 因为生成的代理类 已经继承了 Proxy,Java 单继承,只能通过接口扩展。

相关推荐
乐之者v1 分钟前
AI编程-- codex并行开发需求
java
weixin_520649874 分钟前
C#线程底层原理知识
java·开发语言
AC赳赳老秦8 分钟前
OpenClaw与Excel联动:批量读取/写入数据,生成可视化报表
开发语言·python·excel·产品经理·策略模式·deepseek·openclaw
code_whiter9 分钟前
C++9(vector)
开发语言·c++
xieliyu.10 分钟前
Java手搓数据结构:从零模拟实现单向无头非循环链表
java·数据结构·学习·链表
覆东流10 分钟前
第5天:Python字符串操作进阶
开发语言·后端·python
吴梓穆11 分钟前
UE5 C++ 使C++创建动画蓝图
开发语言·c++·ue5
冰暮流星12 分钟前
javascript之表单事件1
开发语言·前端·javascript
0xDevNull13 分钟前
队列(Queue)实战教程:从原理到架构应用
java·开发语言·后端
ShineWinsu20 分钟前
C++技术文章
开发语言·c++