本文是《深刻字节码 -- 使用 ASM 实现 AOP》的后续博文。在上一篇文章中介绍了如何使用 ASM 动态安插代码到类中,从而简单实现 Aop。文章获得了广大朋友好评,我也但愿能够不负众望继续写出能够获得你们承认的更多相关文章。本文主要讲解 ASM 核心接口方法和其参数意义。另外本文也可用作参考手册使用。 java
ASM 4.0 核心包中包含几个关键类,这些类在ASM 3.0 时期是以接口形式提供。本文针对 ASM 4.0 编写,故不讨论 ASM 3.0环境。首先简单的将 “.class” 文件的内容看做是以下树形结构: 数组
Class Annotation Annotation ... Field Annotation ... Method Annotation ...ASM 在读取 “.class” 文件内容时也会按照递此顺序进行调用。每拜访一个结构中的成员都会使用相应的接口,具体关系以下:
树形关系 | 使用的接口 |
Class | ClassVisitor |
Field | FieldVisitor |
Method | MethodVisitor |
Annotation |
AnnotationVisitor |
ClassVisitor,在 ASM3.0 中是一个接口,到了 ASM4.0 与 ClassAdapter 抽象类合并。主要负责 “拜访” 类成员信息。其中包括(标记在类上的注解,类的构造方法,类的字段,类的方法,静态代码块),它的完整接口以下:
spa
我将重点介绍其中几个关键方法: .net
1.visit(int , int , String , String , String , String[])
该方法是当扫描类时第一个拜访的方法,主要用于类声明使用。下面是对方法中各个参数的示意:visit( 类版本 , 修饰符 , 类名 , 泛型信息 , 继承的父类 , 实现的接口)。例如: 代理
public class TestBean { 等价于: visit(V1_6, ACC_PUBLIC | ACC_SUPER , "org/more/test/asm/simple/TestBean", null, "java/lang/Object", null)
第一个参数:表示类版本:V1_6,表示 “.class” 文件的版本是 JDK 1.6。可用的其余版本有:V1_1(JRE_1.1)、V1_2(J2SE_1.2)、V1_3(J2SE_1.3)、V1_4(J2SE_1.4)、V1_5(J2SE_1.5)、V1_6(JavaSE_1.6)、V1_7(JavaSE_1.7)。咱们所指的 JDK 6 或 JDK 7 实际上就是只 JDK 1.6 或 JDK 1.7。 code
第二个参数:表示类的修饰符:修饰符在 ASM 中是以 “ACC_” 开头的常量进行定义。能够做用到类级别上的修饰符有:ACC_PUBLIC(public)、ACC_PRIVATE(private)、ACC_PROTECTED(protected)、ACC_FINAL(final)、ACC_SUPER(extends)、ACC_INTERFACE(接口)、ACC_ABSTRACT(抽象类)、ACC_ANNOTATION(注解类型)、ACC_ENUM(枚举类型)、ACC_DEPRECATED(标记了@Deprecated注解的类)、ACC_SYNTHETIC。 blog
第三个参数:表示类的名称:一般咱们的类完整类名使用 “org.test.mypackage.MyClass” 来表示,可是到了字节码中会以路径形式表示它们 “org/test/mypackage/MyClass” 值得注意的是虽然是路径表示法可是不须要写明类的 “.class” 扩展名。 继承
第四个参数:表示泛型信息,若是类并未定义任何泛型该参数为空。Java 字节码中表示泛型时分别对接口和类采起不一样的定义。该参数的内容格式以下: 接口
<泛型名:基于的类型....>Ljava/lang/Object; <泛型名::基于的接口....>Ljava/lang/Object;其中 “泛型名:基于的类型” 内容能够无限的写下去,例如:
public class TestBean<T,V,Z> { 泛型参数为:<T:Ljava/lang/Object;V:Ljava/lang/Object;Z:Ljava/lang/Object;>Ljava/lang/Object; 分析结构以下: < T:Ljava/lang/Object; V:Ljava/lang/Object; Z:Ljava/lang/Object; > Ljava/lang/Object;
再或者: ci
public class TestBean<T extends Date, V extends ArrayList> { 泛型参数为:<T:Ljava/util/Date;V:Ljava/util/ArrayList;>Ljava/lang/Object; 分析结构以下: < T:Ljava/util/Date; V:Ljava/util/ArrayList; > Ljava/lang/Object;
以上内容只是针对泛型内容是基于某个具体类型的状况,若是泛型是基于接口而非类型则定义方式会有所不一样,这一点须要注意。例如:
public class TestBean<T extends Serializable, V> { 泛型参数为:<T::Ljava/io/Serializable;V:Ljava/lang/Object;>Ljava/lang/Object; 分析结构以下: < T::Ljava/io/Serializable; //比类型多出一个“:” V:Ljava/lang/Object; > Ljava/lang/Object;
第五个参数:表示所继承的父类。因为 Java 的类是单根结构,即全部类都继承自 java.lang.Object 所以能够简单的理解为任何类都会具备一个父类。虽然在编写 Java 程序时咱们没有去写 extends 关键字去明确继承的父类,可是 JDK在编译时 总会为咱们加上 “ extends Object”。因此假若某一天你看到这样一份代码也不要过于紧张。
第六个参数:表示类实现的接口,在 Java 中类是能够实现多个不一样的接口所以此处是一个数组例如:
public class TestBean implements Serializable , List { 该参数会以 “[java/io/Serializable, java/util/List]” 形式出现。
这里须要补充一些内容,若是类型其自己就是接口类型。对于该方法而言,接口的父类类型是 “java/lang/Object”,接口所继承的全部接口都会出如今第六个参数中。例如:
public inteface TestBean implements Serializable , List { 最后两个参数对应为: "java/lang/Object", ["java/io/Serializable","java/util/List"]
2.visitAnnotation(String , boolean)
该方法是当扫描器扫描到类注解声明时进行调用。下面是对方法中各个参数的示意:visitAnnotation(注解类型 , 注解是否能够在 JVM 中可见)。例如:
@Bean({ "" }) public class TestBean { @Bean等价于: visitAnnotation("Lnet/hasor/core/gift/bean/Bean;", true);下面是 @Bean 的源代码:
@Retention(RetentionPolicy.RUNTIME) @Target({ ElementType.TYPE }) public @interface Bean { /** Bean名称。*/ public String[] value(); }
第一个参数:表示的是,注解的类型。它使用的是(“L” + “类型路径” + “;”)形式表述。
第二个参数:表示的是,该注解是否在 JVM 中可见。这个参数的具体含义能够理解为:若是为 true 表示虚拟机可见,咱们能够经过下面这样的代码获取到注解类型:
testBeanType.getAnnotation(TestAnno.class);
谈到这里就须要额外说明一下在声明注解时常见的 “@Retention(RetentionPolicy.RUNTIME)” 标记。RetentionPolicy 是一个枚举它具有三个枚举元素其每一个含义能够理解为:
1.RetentionPolicy.SOURCE:声明注解只保留在 Java 源程序中,在编译 Java 类时注解信息不会被写入到 Class。若是使用的是这个配置 ASM 也将没法探测到这个注解。
2.RetentionPolicy.CLASS:声明注解仅保留在 Class 文件中,JVM 运行时并不会处理它,这意味着 ASM 能够在 visitAnnotation 时候探测到它,可是经过Class 反射没法获取到注解信息。
3.RetentionPolicy.RUNTIME:这是最经常使用的一种声明,ASM 能够探测到这个注解,同时 Java 反射也能够取得注解的信息。全部用到反射获取的注解都会用到这个配置,就是这个缘由。
3.visitField(int , String , String , String , Object)
该方法是当扫描器扫描到类中字段时进行调用。下面是对方法中各个参数的示意:visitField(修饰符 , 字段名 , 字段类型 , 泛型描述 , 默认值)。例如:
public class TestBean { private String stringData; stringData字段等价于: visitField(ACC_PRIVATE, "stringData", "Ljava/lang/String;", null, null)
第一个参数:表示字段的修饰符,修饰符在 ASM 中是以 “ACC_” 开头的常量进行定义。能够做用到字段级别上的修饰符有:ACC_PUBLIC(public)、ACC_PRIVATE(private)、ACC_PROTECTED(protected)、ACC_STATIC(static)、ACC_FINAL(final)、ACC_VOLATILE(volatile)、ACC_TRANSIENT(transient)、ACC_ENUM(枚举)、ACC_DEPRECATED(标记了@Deprecated注解的字段)、ACC_SYNTHETIC。
第二个参数:表示字段的名称。
第三个参数:表示字段的类型,其格式为:(“L” + 类型路径 + “;”)。
public class TestBean<T, V> { private T data; private V value; 等价于: visit(V1_6, ACC_PUBLIC | ACC_SUPER , "org/more/test/asm/simple/TestBean", "<T:Ljava/lang/Object;V:Ljava/lang/Object;>Ljava/lang/Object;", //定义了两个泛型类型 T 和 V "java/lang/Object", null) visitField(ACC_PRIVATE, "data", "Ljava/lang/Object;", "TT;", null) //data 泛型名称为 T visitField(ACC_PRIVATE, "value", "Ljava/lang/Object;", "TV;", null) // value 泛型名称为 V
public class TestBean<T extends Serializable, V> { private T data; private V value; 等价于: visit(V1_6, ACC_PUBLIC | ACC_SUPER , "org/more/test/asm/simple/TestBean", "<T::Ljava/io/Serializable;V:Ljava/lang/Object;>Ljava/lang/Object;", //定义了两个泛型类型 T 和 V "java/lang/Object", null) visitField(ACC_PRIVATE, "data", "Ljava/io/Serializable;", "TT;", null) //data 泛型名称为 T visitField(ACC_PRIVATE, "value", "Ljava/lang/Object;", "TV;", null) // value 泛型名称为 V第五个参数:表示的是默认值, 因为默认值是 Object 类型你们可能觉得能够是任何类型。这里要澄清一下,默认值中只能用来表述 Java 基本类型这其中包括了(byte、sort、int、long、float、double、boolean、String)其余全部类型都不不能够进行表述。而且只有标有 “final” 修饰符的字段而且该字段赋有初值时这个参数才会有值。例如类:
public class TestBean { private final String data; public TestBean() { data = "aa"; } ....
在执行 “visitField” 方法时候,这个参数的就是 null 值,下面这种代码也会是 null 值:
public class TestBean { private final Date data; public TestBean() { data =new Date(); } ....
此外若是字段使用的是基本类型的包装类型,诸如:Integer、Long...也会为空值:
public class TestBean { private final Integer intData = 12; ...
可以正确获得默认值的代码应该是这个样子的:
public class TestBean { private final String data = "ABC"; private final int intData = 12; ...
4.visitMethod(int , String , String , String , String[])
该方法是当扫描器扫描到类的方法时进行调用。下面是对方法中各个参数的示意:visitMethod(修饰符 , 方法名 , 方法签名 , 泛型信息 , 抛出的异常)。例如:
public class TestBean { public int halloAop(String param) throws Throwable { 等价于: visit(V1_6, ACC_PUBLIC | ACC_SUPER , "org/more/test/asm/simple/TestBean", null, "java/lang/Object", null) visitMethod(ACC_PUBLIC, "<init>", "()V", null, null) visitMethod(ACC_PUBLIC, "halloAop", "(Ljava/lang/String;)I", null, [java/lang/Throwable])
第一个参数:表示方法的修饰符,修饰符在 ASM 中是以 “ACC_” 开头的常量进行定义。能够做用到方法级别上的修饰符有:ACC_PUBLIC(public)、ACC_PRIVATE(private)、ACC_PROTECTED(protected)、ACC_STATIC(static)、ACC_FINAL(final)、ACC_SYNCHRONIZED(同步的)、ACC_VARARGS(不定参数个数的方法)、ACC_NATIVE(native类型方法)、ACC_ABSTRACT(抽象的)、ACC_DEPRECATED(标记了@Deprecated注解的方法)、ACC_STRICT、ACC_SYNTHETIC。
第二个参数:表示方法名,在 ASM 中 “visitMethod” 方法会处理(构造方法、静态代码块、私有方法、受保护的方法、共有方法、native类型方法)。在这些范畴中构造方法的方法名为 “<init>”,静态代码块的方法名为 “<clinit>”。列如:
public class TestBean { public int halloAop(String param) throws Throwable { return 0; } static { System.out.println(); } ... 等价于: visit(V1_6, ACC_PUBLIC | ACC_SUPER , "org/more/test/asm/simple/TestBean", null, "java/lang/Object", null) visitMethod(ACC_PUBLIC, "<clinit>", "()V", null, null) visitMethod(ACC_PUBLIC, "<init>", "()V", null, null) visitMethod(ACC_PUBLIC, "halloAop", "(Ljava/lang/String;)I", null, [java/lang/Throwable])
第三个参数:表示方法签名,方法签名的格式以下:“(参数列表)返回值类型”。在字节码中不一样的类型都有其对应的代码,以下所示:
"I" = int "B" = byte "C" = char "D" = double "F" = float "J" = long "S" = short "Z" = boolean "V" = void "[...;" = 数组 "[[...;" = 二维数组 "[[[...;" = 三维数组 "L....;" = 引用类型
下面是一些方法签名对应的方法参数列表。
String[] |
[Ljava/lang/String; |
String[][] |
[[Ljava/lang/String; |
int, String, String, String, String[] |
ILjava/lang/String;Ljava/lang/String;Ljava/lang/String;[Ljava/lang/String; |
int, boolean, long, String[], double |
IZJ[Ljava/lang/String;D |
Class<?>, String, Object... paramType |
Ljava/lang/Class;Ljava/lang/String;[Ljava/lang/Object; |
int[] |
[I |
第四个参数:凡是具备泛型信息的方法,该参数都会有值。而且该值的内容信息基本等于第三个参数的拷贝,只不过不一样的是泛型参数被特殊标记出来。例如:
public class TestBean<T, V extends List> { public T halloAop(V abc, int aaa) throws Throwable { 方法签名:(Ljava/util/List;I)Ljava/lang/Object; 泛型签名:(TV;I)TT;
public class TestBean<T, V extends List> { public String halloAop(V abc, int aaa) throws Throwable { 方法签名:(Ljava/util/List;I)Ljava/lang/String; 泛型签名:(TV;I)Ljava/lang/String;能够看出泛型信息中用于标识泛型类型的结构是(“T” + 泛型名 + “;”),还有一种状况就是。泛型是声明在方法上。例如:
public class TestBean { public <T extends List> String halloAop(T abc, int aaa) throws Throwable { 方法签名:(Ljava/util/List;I)Ljava/lang/String; 泛型签名:<T::Ljava/util/List;>(TT;I)Ljava/lang/String; //泛型类型基于接口
public class TestBean { public <T> String halloAop(T abc, int aaa) throws Throwable { 方法签名:(Ljava/lang/Object;I)Ljava/lang/String; 泛型签名:<T:Ljava/lang/Object;>(TT;I)Ljava/lang/String; //泛型类型基于类型第五个参数:用来表示将会抛出的异常,若是方法不会抛出异常。则该参数为空。这个参数的表述形式比较简单,举一个例子:
public class TestBean { public <T> String halloAop(T abc, int aaa) throws Throwable,Exception { 异常参数为:[java/lang/Throwable, java/lang/Exception]
5.visitEnd()
该方法是当扫描器完成类扫描时才会调用,若是想在类中追加某些方法。能够在该方法中实现。在后续文章中咱们会用到这个方法。
提示:ACC_SYNTHETIC、ACC_STRICT:这两个修饰符我也不清楚具体功能含义是什么,若是有知道的朋友还望补充说明。
下一篇文章将讲解,使用 ASM 如何实一个 Aop 现代理类,届时将不在详细讲解 ClassVisitor 类中各个方法的做用。