Java內置的注解以及自定義一個注解大家都比較熟悉的了,現(xiàn)在來看看注解實現(xiàn)的原理漫贞,看看Java的體系下面是如何對注解的支持的。
在討論前先看一個自定義注解的例子,自定義實現(xiàn)這樣一個注解:通過@Test向某類注入一個字符串骨杂,通過@TestMethod向某個方法注入一個字符串。
① 創(chuàng)建Test注解雄卷,聲明作用于類并保留到運行時搓蚪,默認值為default。
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)public@interfaceTest {
Stringvalue()default"default";
}
② 創(chuàng)建TestMethod注解丁鹉,聲明作用于方法并保留到運行時妒潭。
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)public@interfaceTestMethod {
Stringvalue();
}
③測試類悴能,運行后輸出default和tomcat-method兩個字符串,因為@Test沒有傳入值雳灾,所以輸出了默認值漠酿,而@TestMethod則輸出了注入的字符串。
@Test()publicclassAnnotationTest {
@TestMethod("tomcat-method")
publicvoidtest(){
}
publicstaticvoidmain(String[] args){
Test t = AnnotationTest.class.getAnnotation(Test.class);
System.out.println(t.value());
TestMethod tm =null;
try{
tm = AnnotationTest.class.getDeclaredMethod("test",null).getAnnotation(TestMethod.class);
}catch(Exception e) {
e.printStackTrace();
}
System.out.println(tm.value());
}
}
對于注解Test谎亩,如果對AnnotationTest類進行注解炒嘲,則運行時可以通過AnnotationTest.class.getAnnotation(Test.class)獲取注解聲明的值,從上面的句子就可以看出匈庭,它是從class結構中獲取出Test注解的夫凸,所以肯定是在某個時候注解被加入到class結構中去了。
@Test("test")publicclassAnnotationTest {
publicvoidtest(){
}
}
從java源碼到class字節(jié)碼是由編譯器完成的阱持,編譯器會對java源碼進行解析并生成class文件寸痢,而注解也是在編譯時由編譯器進行處理,編譯器會對注解符號處理并附加到class結構中紊选,根據(jù)jvm規(guī)范啼止,class文件結構是嚴格有序的格式,唯一可以附加信息到class結構中的方式就是保存到class結構的attributes屬性中兵罢。我們知道對于類献烦、字段、方法卖词,在class結構中都有自己特定的表結構巩那,而且各自都有自己的屬性,而對于注解此蜈,作用的范圍也可以不同即横,可以作用在類上,也可以作用在字段或方法上裆赵,這時編譯器會對應將注解信息存放到類东囚、字段、方法自己的屬性上战授。
在我們的AnnotationTest類被編譯后页藻,在對應的AnnotationTest.class文件中會包含一個RuntimeVisibleAnnotations屬性,由于這個注解是作用在類上植兰,所以此屬性被添加到類的屬性集上份帐。即Test注解的鍵值對value=test會被記錄起來。而當JVM加載AnnotationTest.class文件字節(jié)碼時楣导,就會將RuntimeVisibleAnnotations屬性值保存到AnnotationTest的Class對象中废境,于是就可以通過AnnotationTest.class.getAnnotation(Test.class)獲取到Test注解對象,進而再通過Test注解對象獲取到Test里面的屬性值。
這里可能會有疑問噩凹,Test注解對象是什么朦促?其實注解被編譯后的本質就是一個繼承Annotation接口的接口,所以@Test其實就是public interface Test extends Annotation栓始,當我們通過AnnotationTest.class.getAnnotation(Test.class)調用時务冕,JDK會通過動態(tài)代理生成一個實現(xiàn)了Test接口的對象,并把將RuntimeVisibleAnnotations屬性值設置進此對象中幻赚,此對象即為Test注解對象禀忆,通過它的value()方法就可以獲取到注解值。
Java注解實現(xiàn)機制的整個過程如上面所示落恼,它的實現(xiàn)需要編譯器和JVM一起配合箩退。