net.bytebuddy字节码引擎,动态生成Java类

Net ByteBuddy 是 Java 世界里"运行时动态生成字节码"的一把瑞士军刀------

官方口号:"Code like a Pro, intercept like a Boss."

用纯 API 就能在 "零编译、零源码" 的前提下:

  • 动态创建子类 / 接口实现
  • 拦截(改写)任意方法(包括 final、static、private)
  • 给类加字段、加注解、加接口
  • 生成 Lambda、访问者、代理、AOP 切面、Mock 桩......

而且语法像 fluent-builder,写起来比手写 ASM 舒服 10 倍,性能又比 CGLib / Javassist 高一头;

Spring Framework 5.3+、Mockito 2+、Hibernate 6 都把它当默认字节码引擎。


一句话定位

"ByteBuddy = ASM 的脚手架 + CGLib 的易用性 + Java-Agent 的侵入能力"


核心概念(中文速记)

英文术语 白话解释 对应代码入口
ByteBuddy 工厂大吊车,一切动态类型的起点 new ByteBuddy()
DynamicType 正在"搭建"的类,还没落盘 .subclass(Xxx.class)
ElementMatcher "谁"要被拦截------方法/字段/注解过滤 ElementMatchers.any()
Implementation 拦截后"怎么干"------固定值、调用父类、转发、自定义 MethodDelegation.to(MyInterceptor.class)
Unloaded 已生成字节码但未加载,可 .saveIn() 落盘调试 dynamicType.make()
Loaded 已经进 JVM 的 Class,可直接 newInstance() type.load(getClass().getClassLoader())

10 行代码速通(动态子类 + 方法拦截)

xml 复制代码
		<dependency>
            <groupId>net.bytebuddy</groupId>
            <artifactId>byte-buddy</artifactId>
            <version>1.14.16</version>
        </dependency>
java 复制代码
// 1. 定义拦截器:所有方法都转发到这里
public class TimeInterceptor {
    @RuntimeType
    public Object intercept(@Origin Method method,
                            @SuperCall Callable<?> zuper) throws Exception {
        long start = System.nanoTime();
        try { return zuper.call(); }          // 继续原实现
        finally { System.out.println(method + " 耗时 " + (System.nanoTime()-start)/1_000_000 + " ms"); }
    }
}

// 2. 运行时生成子类并加载
Class<? extends ArrayList> dynamicType = new ByteBuddy()
        .subclass(ArrayList.class)                       // 我要继承 ArrayList
        .method(ElementMatchers.any())                   // 拦截所有方法
        .intercept(MethodDelegation.to(TimeInterceptor.class))
        .make()
        .load(getClass().getClassLoader())
        .getLoaded();

// 3. 像正常类一样用
List<String> list = dynamicType.getDeclaredConstructor().newInstance();
list.add("byte");
list.add("buddy");
System.out.println(list);   // 输出内容 + 每方法耗时

常见玩法地图

  1. 给接口快速生成实现类(MyBatis Mapper 那种)

    java 复制代码
    new ByteBuddy()
        .implement(UserMapper.class)
        .intercept(MethodDelegation.to(sqlSession));
  2. 运行时重新加载已存在的类(Java-Agent 热补丁)

    java 复制代码
    ByteBuddyAgent.install();          // attach 到当前 JVM
    new ByteBuddy()
        .redefine(FooService.class)
        .method(named("calc")).intercept(FixedValue.value(42))
        .installOnByteBuddyAgent();
  3. 生成 Lambda 的"元工厂"字节码,提升反射调用速度(Spring 5.3 内部用它替代 CGLib)。

  4. 单元测试 Mock 桩(Mockito 2 默认底层就是 ByteBuddy)。


性能 & 体积

  • 生成速度:首次约 1~3 ms / 类,之后内部缓存 < 0.1 ms。
  • 运行速度:接近手写子类,无反射开销(生成的字节码里直接 invokespecial / invokevirtual)。
  • 依赖体积:核心 byte-buddy-1.14.x.jar 仅 3.5 MB,无 Native 代码。

一句话总结

"只要你会写 fluent API,就能在运行时'凭空'造出任何 Java 类,并让它按你的规则跑------这就是 ByteBuddy。"

相关推荐
叫我辉哥e11 小时前
新手进阶Python:办公看板集成审批流+精细化权限+日志审计
开发语言·python·信息可视化
JMchen1231 小时前
Android Activity管理工具类
android·java·学习·移动开发·android-studio
pixcarp2 小时前
Golang web工作原理详解
开发语言·后端·学习·http·golang·web
程序员:钧念2 小时前
【sh脚本与Python脚本的区别】
开发语言·人工智能·python·机器学习·语言模型·自然语言处理·transformer
青衫码上行2 小时前
SpringBoot多环境配置
java·spring boot·后端·学习
爬山算法2 小时前
Hibernate(54)Hibernate中的批量更新如何实现?
java·后端·hibernate
Elieal2 小时前
零基础入门 WebSocket:从原理到 Java 实战
java·websocket·网络协议
老毛肚2 小时前
Spring 4.0 Spring MVC。
java·spring·mvc
Pth_you2 小时前
Python权限问题终极解决方案
开发语言·python