小白都能學會的Java註解與反射機制

陳皮的JavaLib發表於2021-04-07

前言

Java註解和反射是很基礎的Java知識了,為何還要講它呢?因為我在面試應聘者的過程中,發現不少面試者很少使用過註解和反射,甚至有人只能說出@Override這一個註解。我建議大家還是儘量能在開發中使用註解和反射,有時候使用它們能讓你事半功倍,簡化程式碼提高編碼的效率。很多優秀的框架都基本使用了註解和反射,在Spring AOP中,就把註解和反射用得淋漓盡致。

什麼是註解

Java註解(Annotation)亦叫Java標註,是JDK5.0開始引入的一種註釋機制。 註解可以用在類、介面,方法、變數、引數以及包等之上。註解可以設定存在於不同的生命週期中,例如SOURCE(原始碼中),CLASS(Class檔案中,預設是此保留級別),RUNTIME(執行期中)。

註解以@註解名的形式存在於程式碼中,Java中內建了一些註解,例如@Override,當然我們也可以自定義註解。註解也可以有引數,例如@MyAnnotation(value = "陳皮")。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}

那註解有什麼作用呢?其一是作為一種輔助資訊,可以對程式做出一些解釋,例如@Override註解作用於方法上,表示此方法是重寫了父類的方法。其二,註解可以被其他程式讀取,例如編譯器,例如編譯器會對被@Override註解的方法檢測判斷方法名和引數等是否與父類相同,否則會編譯報錯;而且在執行期可以通過反射機制訪問某些註解資訊。

內建註解

Java中有10個內建註解,其中6個註解是作用在程式碼上的,4個註解是負責註解其他註解的(即元註解),元註解提供對其他註解的型別說明。

註解 作用 作用範圍
@Override 檢查該方法是否是重寫方法。如果其繼承的父類或者實現的介面中並沒有該方法時,會報編譯錯誤。 作用在程式碼上
@Deprecated 標記表示過時的,不推薦使用。可以用於修飾方法,屬性,類。如果使用被此註解修飾的方法,屬性或類,會報編譯警告。 作用在程式碼上
@SuppressWarnings 告訴編譯器忽略註解中宣告的警告。 作用在程式碼上
@SafeVarargs Java 7開始支援,忽略任何使用引數為泛型變數的方法或建構函式呼叫產生的警告。 作用在程式碼上
@FunctionalInterface Java 8開始支援,標識一個匿名函式或函式式介面。 作用在程式碼上
@Repeatable Java 8開始支援,標識某註解可以在同一個宣告上使用多次。 作用在程式碼上
@Retention 標識這個註解的儲存級別,是隻在程式碼中,還是編入class檔案中,或者是在執行時可以通過反射訪問。包含關係runtime>class>source。 作用在其他註解上,即元註解
@Documented 標記這些註解是否包含在使用者文件中javadoc。 作用在其他註解上,即元註解
@Target 標記某個註解的使用範圍,例如作用方法上,類上,屬性上等等。如果註解未使用@Target,則註解可以用於任何元素上。 作用在其他註解上,即元註解
@Inherited 說明子類可以繼承父類中的此註解,但這不是真的繼承,而是可以讓子類Class物件使用getAnnotations()獲取父類被@Inherited修飾的註解 作用在其他註解上,即元註解

自定義註解

使用@interface關鍵字自定義註解,其實底層就是定義了一個介面,而且自動繼承java.lang.annotation.Annotation介面。

我們自定義一個註解如下:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface MyAnnotation {
    String value();
}

我們使用命令javap反編譯我們定義的MyAnnotation註解的class檔案,結果顯示如下。雖然註解隱式繼承了Annotation介面,但是Java不允許我們顯示通過extends關鍵字繼承Annotation介面甚至其他介面,否則編譯報錯。

D:\>javap MyAnnotation.class
Compiled from "MyAnnotation.java"
public interface com.nobody.MyAnnotation extends java.lang.annotation.Annotation {
  public abstract java.lang.String value();
}

註解的定義內容如下:

  • 格式為public @interface 註解名 {定義內容}
  • 內部的每一個方法實際是宣告瞭一個引數,方法的名稱就是引數的名稱。
  • 返回值型別就是引數的型別,而且返回值型別只能是基本型別(int,float,long,short,boolean,byte,double,char),Class,String,enum,Annotation以及上述型別的陣列形式。
  • 如果定義了引數,可通過default關鍵字宣告引數的預設值,若不指定預設值,使用時就一定要顯示賦值,而且不允許使用null值,一般會使用空字串或者0。
  • 如果只有一個引數,一般引數名為value,因為使用註解時,賦值可以不顯示寫出引數名,直接寫引數值。
import java.lang.annotation.*;

/**
 * @Description 自定義註解
 * @Author Mr.nobody
 * @Date 2021/3/30
 * @Version 1.0
 */
@Target(ElementType.METHOD) // 此註解只能用在方法上。
@Retention(RetentionPolicy.RUNTIME) // 此註解儲存在執行時期,可以通過反射訪問。
@Inherited // 說明子類可以繼承此類的此註解。
@Documented // 此註解包含在使用者文件中。
public @interface CustomAnnotation {
    String value(); // 使用時需要顯示賦值
    int id() default 0; // 有預設值,使用時可以不賦值
}
/**
 * @Description 測試註解
 * @Author Mr.nobody
 * @Date 2021/3/30
 * @Version 1.0
 */
public class TestAnnotation {

    // @CustomAnnotation(value = "test") 只能註解在方法上,這裡會報錯
    private String str = "Hello World!";

    @CustomAnnotation(value = "test")
    public static void main(String[] args) {
        System.out.println(str);
    }
}

Java8 註解

在這裡講解下Java8之後的幾個註解和新特性,其中一個註解是@FunctionalInterface,它作用在介面上,標識是一個函式式介面,即只有有一個抽象方法,但是可以有預設方法。

@FunctionalInterface
public interface Callback<P,R> {

    public R call(P param);
}

還有一個註解是@Repeatable,它允許在同一個位置使用多個相同的註解,而在Java8之前是不允許的。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(OperTypes.class)
public @interface OperType {
    String[] value();
}
// 可以理解@OperTypes註解作為接收同一個型別上重複@OperType註解的容器
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface OperTypes {
    OperType[] value();
}
@OperType("add")
@OperType("update")
public class MyClass {
    
}

注意,對於重複註解,不能再通過clz.getAnnotation(Class<A> annotationClass)方法來獲取重複註解,Java8之後,提供了新的方法來獲取重複註解,即clz.getAnnotationsByType(Class<A> annotationClass)方法。

package com.nobody;

import java.lang.annotation.Annotation;

/**
 * @Description
 * @Author Mr.nobody
 * @Date 2021/3/31
 * @Version 1.0
 */
@OperType("add")
@OperType("update")
public class MyClass {

    public static void main(String[] args) {
        Class<MyClass> clz = MyClass.class;
        Annotation[] annotations = clz.getAnnotations();
        for (Annotation annotation : annotations) {
            System.out.println(annotation.toString());
        }

        OperType operType = clz.getAnnotation(OperType.class);
        System.out.println(operType);

        OperType[] operTypes = clz.getAnnotationsByType(OperType.class);
        for (OperType type : operTypes) {
            System.out.println(type.toString());
        }
    }

}

// 輸出結果為
@com.nobody.OperTypes(value=[@com.nobody.OperType(value=[add]), @com.nobody.OperType(value=[update])])
null
@com.nobody.OperType(value=[add])
@com.nobody.OperType(value=[update])

在Java8中,ElementType列舉新增了兩個列舉成員,分別為TYPE_PARAMETER和TYPE_USE,TYPE_PARAMETER標識註解可以作用於型別引數,TYPE_USE標識註解可以作用於標註任意型別(除了Class)。

Java反射機制

我們先了解下什麼是靜態語言和動態語言。動態語言是指在執行時可以改變其自身結構的語言。例如新的函式,物件,甚至程式碼可以被引進,已有的函式可以被刪除或者結構上的一些變化。簡單說即是在執行時程式碼可以根據某些條件改變自身結構。動態語言主要有C#,Object-C,JavaScript,PHP,Python等。靜態語言是指執行時結構不可改變的語言,例如Java,C,C++等。

Java不是動態語言,但是它可以稱為準動態語言,因為Java可以利用反射機制獲得類似動態語言的特性,Java的動態性讓它在程式設計時更加靈活。

反射機制允許程式在執行期藉助於Reflection API取得任何類的內部資訊,並能直接操作任意物件的內部屬性以及方法等。類在被載入完之後,會在堆記憶體的方法區中生成一個Class型別的物件,一個類只有一個Class物件,這個物件包含了類的結構資訊。我們可以通過這個物件看到類的結構。

比如我們可以通過Class clz = Class.forName("java.lang.String");獲得String類的Class物件。我們知道每個類都隱式繼承Object類,Object類有個getClass()方法也能獲取Class物件。

Java反射機制提供的功能

  1. 在執行時判斷任意一個物件所屬的類
  2. 在執行時構造任意一個類的物件
  3. 在執行時判斷任意一個類具有的成員變數和方法
  4. 在執行時獲取泛型資訊
  5. 在執行時呼叫任意一個物件的成員變數和方法
  6. 在執行時獲取註解
  7. 生成動態代理
  8. ...

Java反射機制的優缺點

  • 優點:實現動態建立物件和編譯,有更加的靈活性。
  • 缺點:對效能有影響。使用反射其實是一種解釋操作,即告訴JVM我們想要做什麼,然後它滿足我們的要求,所以總是慢於直接執行相同的操作。

Java反射相關的主要API

  • java.lang.Class:代表一個類
  • java.lang.reflect.Method:代表類的方法
  • java.lang.reflect.Field:代表類的成員變數
  • java.lang.reflect.Constructor:代表類的構造器

我們知道在執行時通過反射可以準確獲取到註解資訊,其實以上類(Class,Method,Field,Constructor等)都直接或間接實現了AnnotatedElement介面,並實現了它定義的方法,AnnotatedElement介面的作用主要用於表示正在JVM中執行的程式中已使用註解的元素,通過該介面提供的方法可以獲取到註解資訊。

java.lang.Class 類

在Java反射中,最重要的是Class這個類了。Class本身也是一個類。當程式想要使用某個類時,如果此類還未被載入到記憶體中,首先會將類的class檔案位元組碼載入到記憶體中,並將這些靜態資料轉換為方法區的執行時資料結構,然後生成一個Class型別的物件(Class物件只能由系統建立),一個類只有一個Class物件,這個物件包含了類的結構資訊。我們可以通過這個物件看到類的結構。每個類的例項都會記得自己是由哪個Class例項所生成的。

通過Class物件可以知道某個類的屬性,方法,構造器,註解,以及實現了哪些介面等資訊。注意,只有class,interface,enum,annotation,primitive type,void,[] 等才有Class物件。

package com.nobody;

import java.lang.annotation.ElementType;
import java.util.Map;

public class TestClass {

    public static void main(String[] args) {

        // 類
        Class<MyClass> myClassClass = MyClass.class;
        // 介面
        Class<Map> mapClass = Map.class;
        // 列舉
        Class<ElementType> elementTypeClass = ElementType.class;
        // 註解
        Class<Override> overrideClass = Override.class;
        // 原生型別
        Class<Integer> integerClass = Integer.class;
        // 空型別
        Class<Void> voidClass = void.class;
        // 一維陣列
        Class<String[]> aClass = String[].class;
        // 二維陣列
        Class<String[][]> aClass1 = String[][].class;
        // Class類也有Class物件
        Class<Class> classClass = Class.class;

        System.out.println(myClassClass);
        System.out.println(mapClass);
        System.out.println(elementTypeClass);
        System.out.println(overrideClass);
        System.out.println(integerClass);
        System.out.println(voidClass);
        System.out.println(aClass);
        System.out.println(aClass1);
        System.out.println(classClass);
    }
}

// 輸出結果
class com.nobody.MyClass
interface java.util.Map
class java.lang.annotation.ElementType
interface java.lang.Override
class java.lang.Integer
void
class [Ljava.lang.String;
class [[Ljava.lang.String;
class java.lang.Class

獲取Class物件的方法

  1. 如果知道具體的類,可通過類的class屬性獲取,這種方法最安全可靠並且效能最高。Class clz = User.class;
  2. 通過類的例項的getClass()方法獲取。Class clz = user.getClass();
  3. 如果知道一個類的全限定類名,並且在類路徑下,可通過Class.forName()方法獲取,但是可能會丟擲ClassNotFoundException。Class clz = Class.forName("com.nobody.User");
  4. 內建的基本資料型別可以直接通過類名.Type獲取。Class<Integer> clz = Integer.TYPE;
  5. 通過類載入器ClassLoader獲取

Class類的常用方法

  • public static Class<?> forName(String className):建立一個指定全限定類名的Class物件
  • public T newInstance():呼叫Class物件所代表的類的無參構造方法,建立一個例項
  • public String getName():返回Class物件所代表的類的全限定名稱。
  • public String getSimpleName():返回Class物件所代表的類的簡單名稱。
  • public native Class<? super T> getSuperclass():返回Class物件所代表的類的父類的Class物件,這是一個本地方法
  • public Class<?>[] getInterfaces():返回Class物件的介面
  • public Field[] getFields():返回Class物件所代表的實體的public屬性Field物件陣列
  • public Field[] getDeclaredFields():返回Class物件所代表的實體的所有屬性Field物件陣列
  • public Field getDeclaredField(String name):獲取指定屬性名的Field物件
  • public Method[] getDeclaredMethods():返回Class物件所代表的實體的所有Method物件陣列
  • public Method getDeclaredMethod(String name, Class<?>... parameterTypes):返回指定名稱和引數型別的Method物件
  • myClassClass.getDeclaredConstructors();:返回所有Constructor物件的陣列
  • public ClassLoader getClassLoader():返回當前類的類載入器

在反射中經常會使用到Method的invoke方法,即public Object invoke(Object obj, Object... args),我們簡單說明下:

  • 第一個Object對應原方法的返回值,若原方法沒有返回值,則返回null。
  • 第二個Object物件對應呼叫方法的例項,若原方法為靜態方法,則引數obj可為null。
  • 第二個Object對應若原方法形參列表,若引數為空,則引數args為null。
  • 若原方法宣告為private修飾,則呼叫invoke方法前,需要顯示呼叫方法物件的method.setAccessible(true)方法,才可訪問private方法。

反射操作泛型

泛型是JDK 1.5的一項新特性,它的本質是引數化型別(Parameterized Type)的應用,也就是說所操作的資料型別被指定為一個引數,在用到的時候再指定具體的型別。這種引數型別可以用在類、介面和方法的建立中,分別稱為泛型類、泛型介面和泛型方法。

在Java中,採用泛型擦除的機制來引入泛型,泛型能編譯器使用javac時確保資料的安全性和免去強制型別轉換問題,泛型提供了編譯時型別安全檢測機制,該機制允許程式設計師在編譯時檢測到非法的型別。並且一旦編譯完成,所有和泛型有關的型別會被全部擦除。

Java新增了ParameterizedTypeGenericArrayTypeTypeVariableWildcardType等幾種型別,能讓我們通過反射操作這些型別。

  • ParameterizedType:表示一種引數化型別,比如Collection<String>
  • GenericArrayType:表示種元素型別是引數化型別或者型別變數的陣列型別
  • TypeVariable:是各種型別變數的公共父介面
  • WildcardType:代表種萬用字元型別表示式
package com.nobody;

import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Map;

public class TestReflectGenerics {

    public Map<String, Person> test(Map<String, Integer> map, Person person) {
        return null;
    }

    public static void main(String[] args) throws NoSuchMethodException {
        // 獲取test方法物件
        Method test = TestReflectGenerics.class.getDeclaredMethod("test", Map.class, Person.class);
        // 獲取方法test的引數型別
        Type[] genericParameterTypes = test.getGenericParameterTypes();
        for (Type genericParameterType : genericParameterTypes) {
            System.out.println("方法引數型別:" + genericParameterType);
            // 如果引數型別等於引數化型別
            if (genericParameterType instanceof ParameterizedType) {
                // 獲得真實引數型別
                Type[] actualTypeArguments =
                        ((ParameterizedType) genericParameterType).getActualTypeArguments();
                for (Type actualTypeArgument : actualTypeArguments) {
                    System.out.println("    " + actualTypeArgument);
                }
            }
        }

        // 獲取方法test的返回值型別
        Type genericReturnType = test.getGenericReturnType();
        System.out.println("返回值型別:" + genericReturnType);
        // 如果引數型別等於引數化型別
        if (genericReturnType instanceof ParameterizedType) {
            // 獲得真實引數型別
            Type[] actualTypeArguments =
                    ((ParameterizedType) genericReturnType).getActualTypeArguments();
            for (Type actualTypeArgument : actualTypeArguments) {
                System.out.println("    " + actualTypeArgument);
            }
        }

    }
}

class Person {}

// 輸出結果
方法引數型別:java.util.Map<java.lang.String, java.lang.Integer>
    class java.lang.String
    class java.lang.Integer
方法引數型別:class com.nobody.Person
返回值型別:java.util.Map<java.lang.String, com.nobody.Person>
    class java.lang.String
    class com.nobody.Person

反射操作註解

在Java執行時,通過反射獲取程式碼中的註解是比較常用的手段了,獲取到了註解之後,就能知道註解的所有資訊了,然後根據資訊進行相應的操作。下面通過一個例子,獲取類和屬性的註解,解析對映為資料庫中的表資訊。

package com.nobody;

import java.lang.annotation.*;

public class AnalysisAnnotation {

    public static void main(String[] args) throws Exception {
        Class<?> aClass = Class.forName("com.nobody.Book");
        // 獲取類的指定註解,並且獲取註解的值
        Table annotation = aClass.getAnnotation(Table.class);
        String value = annotation.value();
        System.out.println("Book類對映的資料庫表名:" + value);

        java.lang.reflect.Field bookName = aClass.getDeclaredField("bookName");
        TableField annotation1 = bookName.getAnnotation(TableField.class);
        System.out.println("bookName屬性對映的資料庫欄位屬性 - 列名:" + annotation1.colName() + ",型別:"
                + annotation1.type() + ",長度:" + annotation1.length());
        java.lang.reflect.Field price = aClass.getDeclaredField("price");
        TableField annotation2 = price.getAnnotation(TableField.class);
        System.out.println("price屬性對映的資料庫欄位屬性 - 列名:" + annotation2.colName() + ",型別:"
                + annotation2.type() + ",長度:" + annotation2.length());
    }
}

// 作用於類的註解,用於解析表資料
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Table {
    // 表名
    String value();
}

// 作用於欄位,用於解析表列
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface TableField {
    // 列名
    String colName();

    // 列型別
    String type();

    // 長度
    int length();
}

@Table("t_book")
class Book {
    @TableField(colName = "name", type = "varchar", length = 15)
    String bookName;
    @TableField(colName = "price", type = "int", length = 10)
    int price;
}

// 輸出結果
Book類對映的資料庫表名:t_book
bookName屬性對映的資料庫欄位屬性 - 列名:name,型別:varchar,長度:15
price屬性對映的資料庫欄位屬性 - 列名:price,型別:int,長度:10

效能分析

前面我們說過,反射對效能有一定影響。因為反射是一種解釋操作,它總是慢於直接執行相同的操作。而且Method,Field,Constructor都有setAccessible()方法,它的作用是開啟或禁用訪問安全檢查。如果我們程式程式碼中用到了反射,而且此程式碼被頻繁呼叫,為了提高反射效率,則最好禁用訪問安全檢查,即設定為true。

package com.nobody;

import java.lang.reflect.Method;

public class TestReflectSpeed {

    // 10億次
    private static int times = 1000000000;

    public static void main(String[] args) throws Exception {
        test01();
        test02();
        test03();
    }

    public static void test01() {
        Teacher t = new Teacher();
        long start = System.currentTimeMillis();
        for (int i = 0; i < times; i++) {
            t.getName();
        }
        long end = System.currentTimeMillis();
        System.out.println("普通方式執行10億次消耗:" + (end - start) + "ms");
    }

    public static void test02() throws Exception {
        Teacher teacher = new Teacher();
        Class<?> aClass = Class.forName("com.nobody.Teacher");
        Method getName = aClass.getDeclaredMethod("getName");
        long start = System.currentTimeMillis();
        for (int i = 0; i < times; i++) {
            getName.invoke(teacher);
        }
        long end = System.currentTimeMillis();
        System.out.println("反射方式執行10億次消耗:" + (end - start) + "ms");
    }

    public static void test03() throws Exception {
        Teacher teacher = new Teacher();
        Class<?> aClass = Class.forName("com.nobody.Teacher");
        Method getName = aClass.getDeclaredMethod("getName");
        getName.setAccessible(true);
        long start = System.currentTimeMillis();
        for (int i = 0; i < times; i++) {
            getName.invoke(teacher);
        }
        long end = System.currentTimeMillis();
        System.out.println("關閉安全檢查反射方式執行10億次消耗:" + (end - start) + "ms");
    }

}


class Teacher {

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

//輸出結果
普通方式執行10億次消耗:13ms
反射方式執行10億次消耗:20141ms
關閉安全檢查反射方式執行10億次消耗:8233ms

通過實驗可知,反射比直接執行相同的方法慢了很多,特別是當反射的操作被頻繁呼叫時效果更明顯,當然通過關閉安全檢查可以提高一些速度。所以,放射也不應該氾濫成災的,而是適度使用才能發揮最大作用。

相關文章