java注解
什么是注解
Java 注解也就是Annotation是從 Java5 開始引入的新技術(shù)
-
Annotation的作用:
- 不是程序本身苞尝,可以對程序作出解釋
- 可以被其他程序(編譯器等)讀取
-
Annotation的格式:
- 注解以@注釋名在代碼中存在的,可以添加一些數(shù)值,例如SuppressWarnings(value="unchecked")
-
Annotation在里使用?
- 可以附加在package镜硕,class拓萌、method,filed等上面狸捅,相當(dāng)與給他們添加了額外的輔助信息,我們可以通過反射機(jī)制編程實(shí)現(xiàn)對這些元數(shù)據(jù)的訪問
元注解
- 元注解的作用就是負(fù)責(zé)注解其他注解累提,java定義了4個(gè)標(biāo)準(zhǔn)的meta-annotation類型尘喝,被用來提供對其他annotation類型作說明
- 這些類型和它們所支持的類在java.lang.annotation包中可以找到(@Target,@Retention刻恭,@Documented瞧省,@Inherited)
- @Target:用于描述使用范圍(注解在什么地方使用)
- @Retetion:表示需要在什么級別保證該注釋信息,用于描述注解的生命周期(source<class<runtime)
- @Document:英文意思是文檔鳍贾。它的作用是能夠?qū)⒆⒔庵械脑匕?Javadoc 中去鞍匾。
- @Inherited:注解了的注解修飾了一個(gè)父類,如果他的子類沒有被其他注解修飾骑科,則它的子類也繼承了父類的注解
自定義注解
使用@interface自定義注解時(shí)橡淑,自動繼承了java.lang.annotation.Annotation接口
public class Test03 {
//注解可以顯示賦值,如果沒有默認(rèn)值咆爽,一定要給注解賦值
@Myannotation2(name = "aj",schloos = {"機(jī)電學(xué)院"})
public void test(){
}
@MyAnnotation3("")
public void test2(){
}
}
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@interface Myannotation2{
// 注解的參數(shù)梁棠,參數(shù)類型+參數(shù)名
String name() default "";
int age() default 0;
//如果默認(rèn)值為-1 代表不存在
int id() default -1;
String[] schloos() ;
}
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation3{
String value();
}
給代碼加注解其實(shí)就是這么多置森,關(guān)鍵還是我們?nèi)绾稳プx取注解,這就需要用到反射符糊,下面重點(diǎn)介紹java反射
java反射
反射是java被視為動態(tài)語言的關(guān)鍵凫海,反射機(jī)制允許程序在執(zhí)行期借助Reflection API取得任何類的內(nèi)部信息,并能直接操作任意對象內(nèi)部熟悉及方法
Class c = Class.forName("java.lang.String")
加載完類之后男娄,在堆內(nèi)存的方法區(qū)就產(chǎn)生了一個(gè)Class類型的對象(一個(gè)類只有一個(gè)Class對象)行贪,這個(gè)對象就包含了完整的類的結(jié)構(gòu)信息。我們可以通過這個(gè)對象看到類的結(jié)構(gòu)模闲。這個(gè)對象就像一面鏡子建瘫,透過這個(gè)鏡子看到類的結(jié)構(gòu),所以我們稱之為:反射
image
Class類
對于每個(gè)類而言尸折,JRE都為其保留一個(gè)不變的Class類型的對象啰脚,一個(gè)Class對象包含了特定某個(gè)結(jié)構(gòu)的有關(guān)信息。
- Class本身也是一個(gè)類
- Class對象只能由系統(tǒng)建立對象
- 一個(gè)加載的類在jvm中只會有一個(gè)CLass實(shí)例
- 一個(gè)Class對象對應(yīng)的是一個(gè)加載到j(luò)vm中的一個(gè).class文件
- 每個(gè)類的實(shí)例都會記得自己是由哪個(gè)Class實(shí)例生成的
- 通過Class可以完整的得到一個(gè)類中的所有被加載的結(jié)構(gòu)
- Class類是Reflection的根源实夹,針對任何你想動態(tài)加載橄浓、運(yùn)行的類,唯有先獲得相應(yīng)的Class對象
Class類的常用方法
[圖片上傳失敗...(image-44831a-1620553694720)]
反射獲取對象
public class Test02 {
public static void main(String[] args) throws ClassNotFoundException {
Person person = new Student();
System.out.println("這個(gè)人是"+person.name);
//通過對象獲取
Class c1 = person.getClass();
System.out.println(c1.hashCode());
//通過forname獲取
Class c2 = Class.forName("reflection.Student");
System.out.println(c2.hashCode());
//通過類名獲取
Class c3 = Student.class;
System.out.println(c3.hashCode());
//獲得父類類型
Class c4 = c1.getSuperclass();
System.out.println(c4);
}
}
@Data
class Person{
public String name;
public int age;
}
class Student extends Person{
public Student(){
this.name = "學(xué)生";
}
}
class Teacher extends Person{
public Teacher(){
this.name = "老師";
}
}
反射操作方法收擦、屬性
public class Test03 {
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
Class c1 = Class.forName("reflection.Student");
Student student = (Student) c1.newInstance();
System.out.println(student.getName());
// 通過反射操作方法
Method setName = c1.getDeclaredMethod("setName", String.class);
setName.invoke(student, "zhangshan");
System.out.println(student.getName());
Student student1 = (Student) c1.newInstance();
Field name = c1.getDeclaredField("name");
//反射不能直接操作私有屬性贮配,需要手動關(guān)掉程序的安全檢測谍倦,setAccessible(true)
name.setAccessible(true);
name.set(student1,"lisi");
System.out.println(student1.getName());
}
}
性能檢測
public class Test04 {
public static void test01(){
User user = new User();
long startTime = System.currentTimeMillis();
for (int i = 0; i <1000000000 ; i++) {
user.getName();
}
long endTime = System.currentTimeMillis();
System.out.println(endTime - startTime +"ms");
}
public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
User user = new User();
long startTime = System.currentTimeMillis();
Class c1 = user.getClass();
Method getName = c1.getDeclaredMethod("getName", null);
for (int i = 0; i <1000000000 ; i++) {
getName.invoke(user, null);
}
long endTime = System.currentTimeMillis();
System.out.println(endTime - startTime +"ms");
}
public static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
User user = new User();
long startTime = System.currentTimeMillis();
Class c1 = user.getClass();
Method getName = c1.getDeclaredMethod("getName", null);
getName.setAccessible(true);
for (int i = 0; i <1000000000 ; i++) {
getName.invoke(user, null);
}
long endTime = System.currentTimeMillis();
System.out.println(endTime - startTime +"ms");
}
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
test01();
test02();
test03();
}
}
image
反射操作注解
public class Test05 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
Class<?> c1 = Class.forName("reflection.Customer");
// 通過反射獲取注解
Annotation[] annotations = c1.getAnnotations();
for (Annotation annotation:annotations){
System.out.println(annotation);
}
// 獲取注解的值
TableAnnotation annotation = c1.getAnnotation(TableAnnotation.class);
System.out.println(annotation.value());
//獲取類指定注解
Field id = c1.getDeclaredField("id");
FiledAnnotation annotation1 = id.getAnnotation(FiledAnnotation.class);
System.out.println(annotation1.columnName());
System.out.println(annotation1.length());
System.out.println(annotation1.type());
}
}
//類注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface TableAnnotation{
String value();
}
@Data
@TableAnnotation("db_customer")
class Customer {
@FiledAnnotation(columnName="id",type = "Long",length =10)
private Long id;
@FiledAnnotation(columnName="age",type = "int",length =10)
private int age;
@FiledAnnotation(columnName="name",type = "String",length =10)
private String name;
}
//方法注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface FiledAnnotation{
String columnName();
String type();
int length();
}