1.反射獲取對象的變量和方法等信息。
Class c = user.getClass();
Field field = c.getDeclaredField("id");
Method method = c.getDeclaredMethod("say", String.class);
method.invoke(user, "great!");
c 獲取Class對象。
Field getDeclaredField
獲取變量
Method getDeclaredMethod
獲取方法,String.class 指定變量
method.invoke(user, "great!");
//方法執行 帶參數
需要注意的是如果方法是private直接訪問出錯,此時使用method.setAccessible(true);
2.註解:主要是實現該註解相關功能
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.CLASS)
public @interface TestAnno {
String name() default "init name";
}
Target指明註解的作用域:
- TYPE : 類、接口或enum聲明
- FIELD: 域(屬性)聲明
- METHOD: 方法聲明
- PARAMETER: 參數聲明
- CONSTRUCTOR: 構造方法聲明
- LOCAL_VARIABLE:局部變量聲明
- ANNOTATION_TYPE:註釋類型聲明
- PACKAGE: 包聲明
Retention指定註解級別。
- SOURCE:源碼級別,註解只存在源碼中,一般用於和編譯器交互,用於檢測代碼。如@Override, @SuppressWarings。
- CLASS:字節碼級別,註解存在於源碼和字節碼文件中,主要用於編譯時生成額外的文件,如XML,Java文件等,但運行時無法獲得。 如mybatis生成實體和映射文件,這個級別需要添加JVM加載時候的代理(javaagent),使用代理來動態修改字節碼文件。
- RUNTIME:運行時級別,註解存在於源碼、字節碼、java虛擬機中,主要用於運行時,可以使用反射獲取相關的信息。
通過反射獲取註解信息(這個很多框架也是通過這種方式解析註解,進而進行註解功能的開發):
System.out.println(field.getDeclaredAnnotation(TestAnno.class).name());//111
System.out.println(c.isAnnotationPresent(TestAnno.class));//true
TestAnno.class
package com.hiro.learn.anno;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* Project: learn
*
* @author : hiro
* @date : 2020/3/7 22:25
*/
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.CLASS)
public @interface TestAnno {
String name() default "init name";
}
Main.class
package com.hiro.learn.anno;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
/**
* Project: learn
*
* @author : hiro
* @date : 2020/3/7 21:52
*/
public class Main {
public static void main(String[] args) throws NoSuchFieldException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
User user = new User();
user.id = 1;
user.name = "apple";
Class c = user.getClass();
Field field = c.getDeclaredField("id");
Method method = c.getDeclaredMethod("say", String.class);
method.invoke(user, "great!");
}
}
@TestAnno
class User {
String name;
@TestAnno(name = "111")
int id;
@TestAnno
void say(String something) {
System.out.println("user say: " + something);
}
}