我正在尝试在运行时加载类,并在此时用一些AspectJ方面编织它们。我启用了加载时编织,并且当我更常规地使用它时它可以工作。
我的@Aspect类中有以下内容:
@Before("call(* mypackage.MyInterface.*())")
public void myInterfaceExecuteCall(JoinPoint thisJoinPoint,
JoinPoint.StaticPart thisJoinPointStaticPart,
JoinPoint.EnclosingStaticPart thisEnclosingJoinPointStaticPart) {
System.out.println(thisJoinPoint.getSignature().getDeclaringType());
System.out.println(thisJoinPoint.getSignature().getName());
}
然后我正在扫描罐子并找到MyInterface
:
URLClassLoader classLoader = new URLClassLoader(new URL[] { urlOfJar },
ClassLoader.getSystemClassLoader());
WeavingURLClassLoader weaver = new WeavingURLClassLoader(
classLoader);
HashSet<Class<?>> executableClasses = new HashSet<Class<?>>();
for (String name : classNamesInJar) {
try {
Class<?> myImplementation = weaver.loadClass(name);
if (MyInterface.class.isAssignableFrom(myImplementation)) {
executableClasses.add(myImplementation);
}
} catch (Exception e) {
e.printStackTrace();
} catch (NoClassDefFoundError e) {
e.printStackTrace();
}
}
...然后我在某些时候在加载的类中执行一个特定的方法:
try {
Method execute = myImplementation.getMethod("execute");
execute.invoke(myImplementation.newInstance());
} catch (Exception e) {
e.printStackTrace();
}
但是,当我调用execute.invoke(...)
时,我上面给你的@Before方法永远不会执行(虽然execute
方法本身显然已执行,因为我看到它的输出)。
有人知道我做错了什么吗?在调用加载的类的方法之前调用myInterfaceExecuteCall
的方法是什么?
答案 0 :(得分:6)
好的,我发现它是什么,它不能像我预期的那样工作,但这是一个解决方法:
只需@Before("execution(* mypackage.MyInterface.*())")
代替call
。即使在手动加载类并且在运行时由自定义类加载器加载,这也可以工作。这是因为AspectJ不关心使用Method.invoke(...)
完成的调用。我希望其他人可以使用此解决方法。
以下是包含令人难忘的信息的文档的链接:
例如,调用切入点不会选择对java.lang.reflect.Method.invoke(Object,Object [])中实现的方法的反射调用。
http://www.eclipse.org/aspectj/doc/released/progguide/implementation.html
如果您有其他解决方案,请随时回答!
答案 1 :(得分:0)
如果我没弄错的话,AspectJ无法编织JDK类。这可以解释这一点:
例如,调用切入点不会选择对java.lang.reflect.Method.invoke(Object,Object [])中实现的方法的反射调用。