Имам много прост 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. Buildscript за моето приложение е супер прост. Следвах инструкциите в 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
също показва, че файловете са правилно вплетени.
Някакви идеи?
Минимален пример за възпроизвеждане на този проблем
ANDROID_BUILD_TARGET_SDK_VERSION
беше грешен за моята инсталация) успях да компилирам проекта. Имаше и NPE, защото в подпроекта app трябваше да променя параметър ajc на"-log", "weave.log",
, защото вашият локален път не съществуваше на моята машина. Сега трябва ли да стартирам проекта локално, за да възпроизведа проблема? - person kriegaex   schedule 17.11.2014gradle clean installDebug
за инсталиране на вашето устройство/емулатор. - person Flo   schedule 18.11.2014ajc
с плъгин за Java, а дори и да знаех, това няма да реши непременно проблема, пред който съм изправен. - person Flo   schedule 18.11.2014java.lang.NoSuchMethodError
. Както описах, ако преместитеTraceAspect.java
от модулаannotation
към модулаapp
, той магически започва да работи. - person Flo   schedule 18.11.2014