我有一个使用 javac / aspectj 编译器编译的 maven 项目。
我想在编译了 javassist 程序的类上运行,该程序操纵编译的类并向它们添加东西。
我想使用“进程类”阶段来运行我的工具。
我的问题是用 javassist 对“目标/类”中创建的类文件进行迭代的最佳方法是什么,以便之后我可以加载、修复和保存。
另一个要求是也在测试类上运行该工具。
如果有一个开源项目可以做类似的事情,那么看到一个活生生的例子会很棒。
谢谢,
阿夫纳
问问题
3421 次
1 回答
9
我最近偶然发现了同样的问题,我写了一个小的 Maven 插件来在构建期间应用 Javassist 类转换。我已经在https://github.com/drochetti/javassist-maven-plugin上分享了代码
您猜对了,您应该使用流程类阶段,而棘手的部分是类路径设置。经过一些试验和错误后,我设法猜测了 Project、Dependencies 和 Javassist 之间的整个 ClassPath 问题(com.github.drochetti.javassist.maven.JavassistMojo.execute()
如果您想检查解决方案,请参阅代码)。
上面的 GitHub 链接有一些指导方针,但基本上你需要:
1 - 在你的pom.xml上配置插件
<plugin>
<groupId>com.github.drochetti</groupId>
<artifactId>javassist-maven-plugin</artifactId>
<version>1.0.0-SNAPSHOT</version>
<configuration>
<includeTestClasses>false</includeTestClasses>
<transformerClasses>
<transformerClass>com.domain.ToStringTransformer</transformerClass>
</transformerClasses>
</configuration>
<executions>
<execution>
<phase>process-classes</phase>
<goals>
<goal>javassist</goal>
</goals>
</execution>
</executions>
</plugin>
2 - 实施 a ClassTransformer
,这是一个例子:
/**
* Silly transformer, used to hack the toString method.
*/
public class ToStringTransformer extends ClassTransformer {
/**
* We'll only transform subtypes of MyInterface.
*/
@Override
protected boolean filter(CtClass candidateClass) throws Exception {
CtClass myInterface = ClassPool.getDefault().get(MyInterface.class.getName());
return !candidateClass.equals(myInterface) && candidateClass.subtypeOf(myInterface);
}
/**
* Hack the toString() method.
*/
@Override
protected void applyTransformations(CtClass classToTransform) throws Exception {
// Actually you must test if it exists, but it's just an example...
CtMethod toStringMethod = classToTransform.getDeclaredMethod("toString");
classToTransform.removeMethod(toStringMethod);
CtMethod hackedToStringMethod = CtNewMethod.make(
"public String toString() { return \"toString() hacked by Javassist\"; }",
classToTransform);
classToTransform.addMethod(hackedToStringMethod);
}
}
注意:要实现转换器,您需要将插件添加为项目的依赖项,但不要担心,因为它仅在构建时使用,它可以提供范围,这样它就不会成为您的依赖项最终构建。
我希望这会有所帮助!如果您需要进一步的帮助,请告诉我。
丹尼尔
于 2012-12-29T02:58:07.460 回答