问题描述
我正在尝试使用ByteBuddy和Java代理进行一些检测。在其中一个步骤中,我想捕获stacktrace并使用我关心的调用对其进行过滤。让我们想象一个premain函数看起来像这样:
public class SeleniumReporter {
public static void premain(final String agentArgs,final Instrumentation inst) {
new AgentBuilder.Default()
.with(new AgentBuilder.InitializationStrategy.SelfInjection.Eager())
.type(named("org.openqa.selenium.remote.RemoteWebDriver"))
.transform((builder,type,classLoader,module) -> builder
.method(nameStartsWith("findElement")
.and(takesArguments(By.class).or(takesArguments(String.class,String.class)))
.and(isPublic())
)
.intercept(Advice.to(FindElementInterceptor.class))
)
.installOn(inst);
}
}
拦截器采用以下形式:
public class FindElementInterceptor {
@Advice.OnMethodExit
public static void log(@Advice.This RemoteWebDriver driver,@Advice.Origin String method,@Advice.AllArguments Object[] args) {
/*
... Some extra code
*/
final String stackTrace = Arrays.stream(Thread.currentThread().getStackTrace())
.map(t -> String.format("%s:%s",t.getClassName(),t.getMethodName()))
.filter(s -> !s.startsWith("org.codehaus.plexus."))
.filter(s -> !s.startsWith("org.apache.maven."))
.collect(Collectors.joining(";"));
System.out.println(stackTrace);
}
}
运行该代码将抛出 java.lang.NoClassDefFoundError ,因为未加载lambda表达式。因此,我的问题是:我该怎么做才能确保已加载?我当时正在考虑创建一个类,并通过ByteBuddy将其与所有实用程序一起加载。有没有更优雅的方式?
解决方法
Byte Buddy无法将lambda表达式从advice方法复制到目标方法。从技术上讲,Lambda只是在定义该类的类中使用私有方法,而这些方法将对目标类不可用。
使用常规迭代来避免此问题。另外,您需要在单个类中实现表达式,然后使用ClassInjector
将这些类注入目标类加载器。