【发布时间】:2014-11-07 13:30:03
【问题描述】:
我有一个非常简单的 AspectJ 方面(使用@AspectJ),它只打印出一条日志消息。我的目标是为我的 android 应用程序中的代码提供建议。现在,只要我的应用程序源代码中有方面类本身,这些方面就可以正常工作。 将方面移动到不同的模块(java -> .jar 或 android lib -> .aar)后,在我的应用程序中运行建议的代码时,会出现以下运行时异常:
java.lang.NoSuchMethodError: com.xxx.xxx.TraceAspect.aspectOf
基本上我的结构是这样的:
Root
+ app (com.android.application)
- MainActivity (with annotation to be adviced)
+ library (android-library)
- TraceAspect (aspect definition)
从 ajc 编译器中,我可以看到 ajc 编译器选择了我的类并正确地建议它们,所以我真的不知道为什么只要我的源代码中有 @AspectJ 类它就可以工作,但停止工作一旦我将它移动到一个 jar 档案中。
我正在使用 gradle。我的应用程序的构建脚本非常简单。我按照http://fernandocejas.com/2014/08/03/aspect-oriented-programming-in-android/中的说明进行操作
import com.android.build.gradle.LibraryPlugin
import org.aspectj.bridge.IMessage
import org.aspectj.bridge.MessageHandler
import org.aspectj.tools.ajc.Main
buildscript {
repositories {
mavenCentral()
}
dependencies {
classpath 'com.android.tools.build:gradle:0.12.+'
classpath 'org.aspectj:aspectjtools:1.8.1'
}
}
apply plugin: 'com.android.application'
repositories {
mavenCentral()
}
dependencies {
compile 'org.aspectj:aspectjrt:1.8.1'
compile project (':library')
}
android.applicationVariants.all { variant ->
AppPlugin plugin = project.plugins.getPlugin(AppPlugin)
JavaCompile javaCompile = variant.javaCompile
javaCompile.doLast {
String[] args = ["-showWeaveInfo",
"-1.5",
"-XnoInline",
"-inpath", javaCompile.destinationDir.toString(),
"-aspectpath", javaCompile.classpath.asPath,
"-d", javaCompile.destinationDir.toString(),
"-classpath", javaCompile.classpath.asPath,
"-bootclasspath", plugin.project.android.bootClasspath.join(File.pathSeparator)]
MessageHandler handler = new MessageHandler(true);
new Main().run(args, handler)
def log = project.logger
for (IMessage message : handler.getMessages(null, true)) {
switch (message.getKind()) {
case IMessage.ABORT:
case IMessage.ERROR:
case IMessage.FAIL:
log.error message.message, message.thrown
break;
case IMessage.WARNING:
log.warn message.message, message.thrown
break;
case IMessage.INFO:
log.info message.message, message.thrown
break;
case IMessage.DEBUG:
log.debug message.message, message.thrown
break;
}
}
}
}
不确定是否重要,但以防万一,我方面的代码:
@Aspect
public class TraceAspect {
private static final String POINTCUT_METHOD = "execution(@com.xxx.TraceAspect * *(..))";
@Pointcut(POINTCUT_METHOD)
public void annotatedMethod() {}
@Around("annotatedMethod()")
public Object weaveJoinPoint(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("Aspect works...");
return joinPoint.proceed();
}
}
类路径
我还检查了javaCompile.classPath,它正确包含library-classes.jar 和我的app-classes.jar。将-log file 添加到ajc 任务也表明文件已正确编织。
有什么想法吗?
重现此问题的最小示例
【问题讨论】:
-
我想帮助 AspectJ 编译器部分,但从未使用过 Gradle,只有 Maven。你能通过在 GitHub 上发布一个最小的、完全自洽的项目版本(可能只有几个虚拟类)来重现问题吗?
-
谢谢!我更新了问题,请查看。
-
抱歉,我无法构建它。看来我需要安装 Android ADK 包括 SDK,我想我已经安装了,即使它花了很长时间。但不知何故,它没有找到或者是错误的版本。通常,Maven 构建会正确解析和下载其所有依赖项,这在 Gradle 中是否有所不同?你能以某种方式为我创建可复制的构建,而不需要所有的 Android 东西,只专注于普通的 Java + AspectJ?
-
好的,经过大量试验和错误(
ANDROID_BUILD_TARGET_SDK_VERSION我的安装错误)我能够编译该项目。还有一个 NPE,因为在 app 子项目中,我必须将 ajc 参数更改为"-log", "weave.log",,因为您的本地路径在我的机器上不存在。现在我是否应该在本地运行项目以重现问题? -
我现在可以重现该问题,但我是一个 Gradle 菜鸟和一个 Android 菜鸟。按照安迪的说法,我可以说的是你用一个普通的 Java 编译器编译你的方面,即它还没有“完成”。然后不知何故,您的构建过程最终将 annotation.jar 中未完成的方面放入 APK。我建议您也只使用 Ajc 构建您的注释子项目,那么您应该没问题。另一种方法是确保已完成而不是未完成的方面类进入 APK。为此,请在 inpath 上获取 annotation.jar。
标签: java android gradle aspectj android-gradle-plugin