4

我正在使用 AspectJ 1.8.8 编译时编织,我有一个这样的块

@SomeAnnotation(value="someValue")
public List doSomething(String someArg) {
    ...
}

where@SomeAnnotation是通过“Around”建议实现的。

用 JD-GUI 查看字节码,我看到以下生成的代码(稍微格式化):

public class SomeClass {
  private static Annotation ajc$anno$5;

  ...

  @SomeAnnotation(value="someValue")
  public List doSomething(String someArg)
  {
    String str = someArg;
    JoinPoint localJoinPoint = Factory.makeJP(ajc$tjp_5, this, this, str);
    Object[] arrayOfObject = new Object[3];
    arrayOfObject[0] = this;
    arrayOfObject[1] = str;
    arrayOfObject[2] = localJoinPoint;
    Annotation tmp56_53 = ajc$anno$5;
    if (tmp56_53 == null) {
      tmp56_53;
    }
    return (List)new SomeClass.AjcClosure11(arrayOfObject).linkClosureAndJoinPoint(69648).around(tmp56_53, (SomeAnnotation)(ajc$anno$5 = SomeClass.class.getDeclaredMethod("doSomething", new Class[] { String.class }).getAnnotation(SomeAnnotation.class)));
  }
}

我想知道为什么那个条件 ( if (tmp56_53...)) 甚至存在,因为它似乎什么都不做(而且在语法上也是不正确的 Java?也许是因为这是由 ajc 生成的?)。我对此感到好奇,因为它会导致覆盖工具(JaCoCo)中的“分支未命中”。


编辑 1

这是来自 javap 的原始 Java 机器代码:

       0: aload_1
       1: astore_2
       2: getstatic     #480                // Field ajc$tjp_10:Lorg/aspectj/lang/JoinPoint$StaticPart;
       5: aload_0
       6: aload_0
       7: aload_2
       8: invokestatic  #312                // Method org/aspectj/runtime/reflect/Factory.makeJP:(Lorg/aspectj/lang/JoinPoint$StaticPart;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Lorg/aspectj/lang/JoinPoint;
      11: astore_3
      12: invokestatic  #339                // Method com/foo/SomeAspect.aspectOf:()Lcom/foo/SomeAspect;
      15: iconst_3
      16: anewarray     #2                  // class java/lang/Object
      19: astore        4
      21: aload         4
      23: iconst_0
      24: aload_0
      25: aastore
      26: aload         4
      28: iconst_1
      29: aload_2
      30: aastore
      31: aload         4
      33: iconst_2
      34: aload_3
      35: aastore
      36: new           #484                // class com/foo/SomeClass$AjcClosure21
      39: dup
      40: aload         4
      42: invokespecial #485                // Method com/foo/SomeClass$AjcClosure21."<init>":([Ljava/lang/Object;)V
      45: ldc_w         #327                // int 69648
      48: invokevirtual #333                // Method org/aspectj/runtime/internal/AroundClosure.linkClosureAndJoinPoint:(I)Lorg/aspectj/lang/ProceedingJoinPoint;
      51: getstatic     #488                // Field ajc$anno$10:Ljava/lang/annotation/Annotation;
      54: dup
      55: ifnonnull     86
      58: pop
      59: ldc           #75                 // class com/foo/SomeClass
      61: ldc_w         #489                // String someArg
      64: iconst_1
      65: anewarray     #348                // class java/lang/Class
      68: dup
      69: iconst_0
      70: ldc           #171                // class java/lang/String
      72: aastore
      73: invokevirtual #352                // Method java/lang/Class.getDeclaredMethod:(Ljava/lang/String;[Ljava/lang/Class;)Ljava/lang/reflect/Method;
      76: ldc_w         #341                // class com/foo/SomeAnnotation
      79: invokevirtual #358                // Method java/lang/reflect/Method.getAnnotation:(Ljava/lang/Class;)Ljava/lang/annotation/Annotation;
      82: dup
      83: putstatic     #488                // Field ajc$anno$10:Ljava/lang/annotation/Annotation;
      86: nop
      87: checkcast     #341                // class com/foo/SomeAnnotation
      90: invokevirtual #362                // Method com/foo/SomeAspect.around:(Lorg/aspectj/lang/ProceedingJoinPoint;Lcom/foo/SomeAnnotation;)Ljava/lang/Object;
      93: pop
      94: return

看起来ifnonnull可能是有条件的,但我对 JVM 指令一点也不熟悉,我仍然不知道为什么 AspectJ 会生成这样的逻辑。

4

1 回答 1

1

tl; dr:这是一个正常的延迟初始化,jd只是很困惑。

字节 16 是它创建的地方new Object[3]

16: anewarray     #2                  // class java/lang/Object

之后你可以在 19-35 中看到它只是立即将局部变量复制到堆栈中(iconst用于索引,aload用于引用),然后将它们写入数组(aastore)。紧接着的下一个字节是 36,它是new操作符(只是分配,然后紧接着是invokespecial运行构造函数)。

这将我们带到字节 48,它调用linkClosureAndJoinPoint. 你没有包括你的常量表,但在 45 处ldc_w #327加载常量值 69648,所以这将我们带到.around.

现在在第 51 字节发生了一些有趣的事情。重建的单链调用jd现在被中断了。字节码将静态注释字段ajc$anno$10(不是 5,jd如前所述)加载到堆栈上。如果该注释字段不为空 (55),则执行跳转到 86(无操作,用作跳转的“着陆点”),执行该转换检查 ( (SomeAnnotation)),然后最终实际调用通知。

跳过的代码 (58-82) 说明了这一点,您将从反编译中识别出:

SomeClass.class
    .getDeclaredMethod("doSomething", new Class[] { String.class })
    .getAnnotation(SomeAnnotation.class)

然后字节 83 将结果存储到静态字段中,并从那里继续执行。

在 Java 术语中,这正是正在发生的事情:

if (cachedAnnotation == null) {
    cachedAnnotation = getAnnotationOnMethodUsingReflection();
}

AspectJ 的字节码在这里非常紧凑和干净(可能是手动优化的,因为这可能是非常热门的代码)。无论是因为这个还是因为该逻辑中断了链接的方法调用,jd都会变得混乱并拆分空检查和分配。

于 2019-02-09T04:34:07.170 回答