ByteBuddy快速实现AOP

1、定义一个Log注解

@Retention(RetentionPolicy.RUNTIME)
public @interface Log {
}

2、定义一个Service

public class Service {
    @Log
    public int foo(int value) {
        System.out.println("foo: " + value);
        return value;
    }

    public int bar(int value) {
        System.out.println("bar: " + value);
        return value;
    }
}

3、通过byteBuddy对service实现字节码增强

public static void main(String[] args) throws Exception {
        Service service = new ByteBuddy()
        		// 定义父类
                .subclass(Service.class)
                // 定义要拦截的方法
                .method(ElementMatchers.any())
                // 对拦截的方法动态增强,增强逻辑LoggerAdvisor
                .intercept(Advice.to(LoggerAdvisor.class))
                // 编译生成字节码
                .make()
                // 通过类加载器加载(AppClassLoader)
                .load(Service.class.getClassLoader())
                // 获取class对象
                .getLoaded()
                .newInstance();
        service.bar(111);
        service.foo(222);
    }

4、增强逻辑

@RuntimeType 注解:
告诉 Byte Buddy 不要进行严格的参数类型检测,在参数匹配失败时,尝试使用类型转换方式(runtime type casting)进行类型转换,匹配相应方法。

@This 注解:
注入被拦截的目标对象。

@AllArguments 注解:
注入目标方法的全部参数,是不是感觉与 Java 反射的那套 API 有点类似了?

@Origin 注解:
注入目标方法对应的 Method 对象。如果拦截的是字段的话,该注解应该标注到 Field 类型参数。

@Super 注解:
注入目标对象。通过该对象可以调用目标对象的所有方法。

@SuperCall:
这个注解比较特殊,我们要在 intercept() 方法中调用目标方法的话,需要通过这种方式注入,

@SuperCall与 Spring AOP 中的 ProceedingJoinPoint.proceed() 方法有点类似,需要注意的是,这里不能修改调用参数,从上面的示例的调用也能看出来,参数不用单独传递,都包含在其中了。

class LoggerAdvisor {
    @Advice.OnMethodEnter
    public static void onMethodEnter(@Advice.Origin Method method, @Advice.AllArguments Object[] arguments) {
        if (method.getAnnotation(Log.class) != null) {
            System.out.println("Enter " + method.getName() + " with arguments: " + Arrays.toString(arguments));
        }
    }

    @Advice.OnMethodExit
    public static void onMethodExit(@Advice.Origin Method method, @Advice.AllArguments Object[] arguments, @Advice.Return Object ret) {
        if (method.getAnnotation(Log.class) != null) {
            System.out.println("Exit " + method.getName() + " with arguments: " + Arrays.toString(arguments) + " return: " + ret);
        }
    }
}

5、执行结果

bar: 111
Enter foo with arguments: [111]
foo: 222
Exit foo with arguments: [222] return: 222
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
ByteBuddy是一个可以在运行时动态生成Java类的类库,它可以创建任意类,不限于实现用于创建运行时代理的接口。使用ByteBuddy可以方便地使用Java代理或在构建过程中手动更改类。它提供了一个方便的API,可以用于实现AOP(面向切面编程)功能。在使用ByteBuddy实现AOP时,可以通过定义注解和拦截器来对目标方法进行增强。首先,需要定义一个注解,比如`@Log`,然后在目标类的方法上添加该注解。接下来,使用ByteBuddy对目标类进行字节码增强,通过拦截器对被注解的方法进行增强逻辑的定义。最后,通过类加载器加载增强后的类,并创建实例进行调用。这样就实现了基于ByteBuddyAOP功能。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* [java agent+bytebuddyaop实现监控、性能检测、日志记录等](https://blog.csdn.net/qq_17589253/article/details/118364827)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [ByteBuddy快速实现AOP](https://blog.csdn.net/qq_44787816/article/details/127271772)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

白鸽呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值