假设我想将 应用于@Advice.OnMethodEnter中声明的方法org.springframework.web.context.support.GenericWebApplicationContext。为此,我编写了这个最小代理:public class SequenceAgent { public static void premain(final String args, final Instrumentation instrumentation) { new AgentBuilder.Default() .with(new AgentBuilder.InitializationStrategy.SelfInjection.Eager()) .type(nameStartsWith( "org.springframework.web.context.support.GenericWebApplicationContext")) .transform((builder, typeDescription, classLoader, module) -> builder .method(any()).intercept(Advice.to(SequenceAdvice.class))) .installOn(instrumentation); } public static class SequenceAdvice { @Advice.OnMethodEnter static void enter(@Advice.This Object thiz, @Advice.Origin Method method, @Advice.AllArguments Object... args) { String className = thiz.getClass().getName(); String methodName = method.getName(); System.out.println("Entered: " + className + "#" + methodName); } }}我希望此配置被过滤掉org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext,因为它不匹配org.springframework.web.context.support.GenericWebApplicationContext,但看起来对此类对象的方法调用也被拦截:import org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext;public class AgentTest { public static void main(String[] args) { AnnotationConfigServletWebServerApplicationContext context = new AnnotationConfigServletWebServerApplicationContext(); context.containsBean("SomeBean"); }}当附加到代理并运行时,它会打印出来(为了便于阅读而包装):Entered: org.springframework.boot.web.servlet.context. AnnotationConfigServletWebServerApplicationContext #getResourcePatternResolver...Entered: org.springframework.boot.web.servlet.context. AnnotationConfigServletWebServerApplicationContext #getResourceCache
1 回答
一只甜甜圈
TA贡献1836条经验 获得超5个赞
字节好友使用 String.startsWith。
您所看到的是由于 Byte Buddy 检测类,而不是实例。在某种程度上,想想 Byte Buddy 将通知代码复制到目标方法中。
结果,所有子类都会受到影响。为了做你正在做的事情,你需要在调用期间检查实例类的类型,就像你想在 Java 中实现它一样。
添加回答
举报
0/150
提交
取消