成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

我知道你用過Annotation,那你知道它的底層實現嗎

開發 前端
用于定義注解的注解,通常用于注解的定義上,標明該注解的使用范圍、生效范圍等。簡而言之,元注解是用來修飾注解的。

概述

注解想必大家在項目中經常使用,比如Spring框架中常用的一些注解:@Controller、@Service、@RequestMapping等等,它是JDK1.5及以后版本引入的一個特性,與類、接口、枚舉是在同一個層次。它可以聲明在包、類、字段、方法、局部變量、方法參數等的前面,用來對這些元素進行說明,注釋。

注解的分類

根據注解的使用場景,主要分為三類,元注解、內置注解和自定義注解。

元注解

用于定義注解的注解,通常用于注解的定義上,標明該注解的使用范圍、生效范圍等。簡而言之,元注解是用來修飾注解的。

@Retention

指定注解信息保留到哪個階段,分別為源代碼階段、編譯Class階段、運行階段。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Retention {
/**
* Returns the retention policy.
* @return the retention policy
*/
RetentionPolicy value();
}
復制代碼
  • SOURCE:保留在源代碼java中,被編譯器丟棄,也就是說在class文件中不包含注解信息,通常用來標記源碼,引起大家的注意,比如自定義一個注解例如@ThreadSafe,用來標識一個類時線程安全的。
  • CLASS:編譯后的class文件中包含注解信息,但是會被jvm丟棄
  • RUNTIME:注解信息在運行期(JVM)保留(.class也有),可以通過反射機制讀取注解的信息

@Target

指定注解的使用范圍,如類、方法、屬性、局部屬性、參數等, 可以多選。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Target {
/**
* Returns an array of the kinds of elements an annotation type
* can be applied to.
* @return an array of the kinds of elements an annotation type
* can be applied to
*/
ElementType[] value();
}
復制代碼

具體可選的枚舉如下:

public enum ElementType {
/** 適用范圍:類、接口、注解類型,枚舉類型enum */
TYPE,
/** 作用于類屬性 (includes enum constants) */
FIELD,
/** 作用于方法 */
METHOD,
/** 作用于參數聲明 */
PARAMETER,
/** 作用于構造函數聲明 */
CONSTRUCTOR,
/** 作用于局部變量聲明 */
LOCAL_VARIABLE,
/** 作用于注解聲明 */
ANNOTATION_TYPE,
/** 作用于包聲明 */
PACKAGE,
/** 作用于類型參數(泛型參數)聲明 */
TYPE_PARAMETER,
/** 作用于使用類型的任意語句(不包括class) */
TYPE_USE
}
復制代碼

@Inherited

加上該注解的注解,表示可以被標注的類子類繼承,比如A上標記了帶有@Inherited 的注解,那么類B繼承了A, 那么B也會有這個注解,默認情況下注解是不支持繼承的。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Inherited {
}
復制代碼

@Document

將此注解包含在 javadoc 中 ,它代表著此注解會被javadoc工具提取成文檔。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Documented {
}
復制代碼

@Repeatable

1.8中加入的元注解,用來標記是否可以重復標記。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Repeatable {
/**
* Indicates the <em>containing annotation type</em> for the
* repeatable annotation type.
* @return
Class<? extends Annotation> value();
}
復制代碼

內置注解

java提供了一些內置注解,可以配合編譯器來檢查代碼的正確性, 我們可以關注他們的元注解。

@Override

標記當前方法是覆寫父類的方法。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}
復制代碼

@Deprecated

標記一個元素為已過期,不要在使用了

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, PARAMETER, TYPE})
public @interface Deprecated {
}
復制代碼

@SuppressWarnings

用來關閉編譯器輸出的警告信息

@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE})
@Retention(RetentionPolicy.SOURCE)
public @interface SuppressWarnings {
String[] value();
}
復制代碼

@FunctionalInterface

java8中引入,標記是一個函數式接口,也就是說有且只有一個抽象方法的接口

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface FunctionalInterface {}
復制代碼

自定義注解

注解遵循的格式一般如下:

//元注解部分 xxxx
@Retention(xxxx)
@Target(xxxx)
public @interface 注解名 {
返回值 屬性名() 默認值;
返回值 屬性名() 默認值;
}
復制代碼
  • 返回值支持的類型如下:java的8種基礎類型(不支持包裝類型)、String、Class、Enum、Annotation、以及上面類型的數組。
  • 默認值可選,非必有。

舉個項目中自定義的栗子:

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface DataAuthorize {
/**
* 資產ID
* @return
*/
String assetId();

/**
* 資產類型
* @return
*/
String assetType();

/**
* 權限代碼
* @return
*/
String authCode() default "";

/**
* 使用的類型
* @return
*/
Class[] useType();
}
復制代碼

使用反射操作注解

大部分情況下,我們的項目或者開源框架中都定義了大量的注解,而且都是@Retention(RetentionPolicy.RUNTIME)運行時階段,我們可以通過反射獲取注解中的信息,所以整體遵循下面的一個范式。

  1. 自定義注解
  2. 掃描注解
  3. 通過反射獲取注解的信息,執行相應的邏輯。

下面我們重點使用下如何用反射來獲取注解的信息。

  1. 定義target是注解的注解
@Inherited
@Retention( value = RetentionPolicy.RUNTIME)
@Target(value = {ElementType.ANNOTATION_TYPE})
public @interface AnnoTest {

String value() default "anno";

}
復制代碼
  1. 定義一個幾乎全量信息的注解
@AnnoTest("alvinAnno")
@Inherited
@Retention( value = RetentionPolicy.RUNTIME)
@Target(value = {ElementType.TYPE_USE,ElementType.PACKAGE,ElementType.FIELD,
ElementType.TYPE_PARAMETER,ElementType.CONSTRUCTOR,ElementType.LOCAL_VARIABLE})
@Documented
public @interface FullAnnoTest {

String value() default "FullAnnoTest";

}
復制代碼
  1. 定義測試類和反射代碼
@FullAnnoTest("package")
package com.alvin.java.core.anno;

public class ParentObj {
}



@FullAnnoTest("testAnnoReflect")
public class TestAnnoReflect<@FullAnnoTest("parameter") T > extends @FullAnnoTest("parent")ParentObj {

@FullAnnoTest("constructor")
TestAnnoReflect() {
}

//注解字段域
private @FullAnnoTest("name") String name;
//注解泛型字段域
private @FullAnnoTest("value") T value;
//注解通配符
private @FullAnnoTest("list") List<@FullAnnoTest("generic") ?> list;
//注解方法
@FullAnnoTest("method") //注解方法參數
public String hello(@FullAnnoTest("methodParameter") String name)
throws @FullAnnoTest("Exception") Exception { // 注解拋出異常
//注解局部變量,現在運行時暫時無法獲取(忽略)
@FullAnnoTest("result") String result;
result = "siting";
System.out.println(name);
return result;
}

public static void main(String[] args) throws Exception {

TestAnnoReflect<String> TestAnnoReflect = new TestAnnoReflect<> ();
Class<TestAnnoReflect<Object>> clazz = (Class<TestAnnoReflect<Object>>) TestAnnoReflect.getClass();
//class的注解
Annotation[] annotations = clazz.getAnnotations();
FullAnnoTest testTmp = (FullAnnoTest) annotations[0];
System.out.println("修飾TestAnnoReflect.class注解value: "+testTmp.value());
//構造器的注解
Constructor<TestAnnoReflect<Object>> constructor = (Constructor<TestAnnoReflect<Object>>) clazz.getDeclaredConstructors()[0];
testTmp = constructor.getAnnotation(FullAnnoTest.class);
System.out.println("修飾構造器的注解value: "+testTmp.value());
//繼承父類的注解
AnnotatedType annotatedType = clazz.getAnnotatedSuperclass();
testTmp = annotatedType.getAnnotation(FullAnnoTest.class);
System.out.println("修飾繼承父類的注解value: "+testTmp.value());
//注解的注解
AnnoTest AnnoTest = testTmp.annotationType().getAnnotation(AnnoTest.class);
System.out.println("修飾注解的注解AnnoTest-value: "+AnnoTest.value());
//泛型參數 T 的注解
TypeVariable<Class<TestAnnoReflect<Object>>> variable = clazz.getTypeParameters()[0];
testTmp = variable.getAnnotation(FullAnnoTest.class);
System.out.println("修飾泛型參數T注解value: "+testTmp.value());
//普通字段域 的注解
Field[] fields = clazz.getDeclaredFields();
Field nameField = fields[0];
testTmp = nameField.getAnnotation(FullAnnoTest.class);
System.out.println("修飾普通字段域name注解value: "+testTmp.value());
//泛型字段域 的注解
Field valueField = fields[1];
testTmp = valueField.getAnnotation(FullAnnoTest.class);
System.out.println("修飾泛型字段T注解value: "+testTmp.value());
//通配符字段域 的注解
Field listField = fields[2];
AnnotatedParameterizedType annotatedPType = (AnnotatedParameterizedType)listField.getAnnotatedType();
testTmp = annotatedPType.getAnnotation(FullAnnoTest.class);
System.out.println("修飾泛型注解value: "+testTmp.value());
//通配符注解 的注解
AnnotatedType[] annotatedTypes = annotatedPType.getAnnotatedActualTypeArguments();
AnnotatedWildcardType annotatedWildcardType = (AnnotatedWildcardType) annotatedTypes[0];
testTmp = annotatedWildcardType.getAnnotation(FullAnnoTest.class);
System.out.println("修飾通配符注解value: "+testTmp.value());
//方法的注解
Method method = clazz.getDeclaredMethod("hello", String.class);
annotatedType = method.getAnnotatedReturnType();
testTmp = annotatedType.getAnnotation(FullAnnoTest.class);
System.out.println("修飾方法的注解value: "+testTmp.value());
//異常的注解
annotatedTypes = method.getAnnotatedExceptionTypes();
testTmp = annotatedTypes[0].getAnnotation(FullAnnoTest.class);
System.out.println("修飾方法拋出錯誤的注解value: "+testTmp.value());
//方法參數的注解
annotatedTypes = method.getAnnotatedParameterTypes();
testTmp = annotatedTypes[0].getAnnotation(FullAnnoTest.class);
System.out.println("修飾方法參數注解value: "+testTmp.value());
//包的注解
Package p = Package.getPackage("com.alvin.java.core.anno");
testTmp = p.getAnnotation(FullAnnoTest.class);
System.out.println("修飾package注解value: "+testTmp.value());
TestAnnoReflect.hello("hello");
}

}
復制代碼
  1. 查看對應的執行結果
修飾TestAnnoReflect.class注解value: testAnnoReflect
修飾構造器的注解value: constructor
修飾繼承父類的注解value: parent
修飾注解的注解AnnoTest-value: alvinAnno
修飾泛型參數T注解value: parameter
修飾普通字段域name注解value: name
修飾泛型字段T注解value: value
修飾泛型注解value: list
修飾通配符注解value: generic
修飾方法的注解value: method
修飾方法拋出錯誤的注解value: Exception
修飾方法參數注解value: methodParameter
修飾package注解value: package
hello
復制代碼

注解的本質和底層實現

大家有沒有想過注解的本質是什么?

我們先通過反編譯查看注解生成的字節碼,可以通過javap -v FullAnnoTest.class查看如下:

可以看到,我們的注解是繼承自Annotation接口。

public interface Annotation {
boolean equals(Object obj);

int hashCode();

String toString();

/**
* Returns the annotation type of this annotation.
* @return
Class<? extends Annotation> annotationType();
}
復制代碼

所以注解相當于一個語法糖一樣,可以方便我們使用,本質上它是繼承自Annotation的一個接口。

那大家有沒有想過它的實現類在哪里?比如下面的代碼,獲取到注解,按照上面的解釋,它是一個接口,那調用value()方法時,它具體調用的哪個實現類呢?我們并沒有寫實現類啊.....

答案當然就是動態代理生成的實現類。

AnnoTest annoTest = testTmp.annotationType().getAnnotation(AnnoTest.class);
System.out.println("修飾注解的注解AnnoTest-value: "+annoTest.value());
復制代碼

我們可以在啟動參數添加如下命令可以查看生成的代理類:
-Dsun.misc.ProxyGenerator.saveGeneratedFiles=true

執行后,生成代理類如下,

代理大致的代碼如下:

public final class $Proxy2 extends Proxy implements FullAnnoTest {
private static Method m1;
private static Method m2;
private static Method m4;
private static Method m0;
private static Method m3;

public $Proxy2(InvocationHandler var1) throws {
super(var1);
}

public final boolean equals(Object var1) throws {
try {
return (Boolean)super.h.invoke(this, m1, new Object[]{var1});
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
}



public final Class annotationType() throws {
try {
return (Class)super.h.invoke(this, m4, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}

public final String value() throws {
try {
return (String)super.h.invoke(this, m3, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}

static {
try {
m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
m2 = Class.forName("java.lang.Object").getMethod("toString");
m4 = Class.forName("com.alvin.java.core.anno.FullAnnoTest").getMethod("annotationType");
m0 = Class.forName("java.lang.Object").getMethod("hashCode");
m3 = Class.forName("com.alvin.java.core.anno.FullAnnoTest").getMethod("value");
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}
復制代碼

我們看value()方法,這里調用了super.h對象,也就是InvocationHandler對象,而我們注解用的是AnnotationInvocationHandler這個子類,我們在invoke方法中打個斷點,就明白了~~

責任編輯:武曉燕 來源: 今日頭條
相關推薦

2019-03-27 14:20:27

大數據核心價值數據分析

2023-10-10 14:03:47

swap排序解法

2023-01-28 10:27:04

2022-04-26 09:01:45

運算符TypeScript代碼

2023-01-31 09:02:24

JSVMVR

2020-08-13 09:55:37

Stream代碼Java

2020-05-15 14:34:16

C語言丹尼斯 · 里奇開發者

2018-08-20 20:46:07

2021-02-26 08:46:46

PHY寄存器網絡

2022-06-01 07:10:43

遞歸字典極限

2020-09-19 17:44:32

Linux計算器命令

2020-06-15 09:41:47

網絡安全數據技術

2018-01-10 08:27:00

2010-11-23 10:21:53

跳槽

2022-02-15 20:08:41

JDKJavaWindows

2019-04-15 10:45:13

pingICMP協議

2022-09-22 14:55:31

前端JavaScripthis

2022-09-26 13:10:17

JavaScriptthis

2022-10-24 09:57:02

runeGo語言

2023-12-12 08:41:01

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产电影一区二区三区爱妃记 | 99re在线视频 | 成人午夜电影网 | 91pron在线| 成在线人视频免费视频 | 午夜一级大片 | 日韩欧美高清dvd碟片 | av国产精品 | 亚洲日本一区二区 | 欧美午夜影院 | 91久久国产综合久久91精品网站 | 亚洲成人自拍网 | 亚洲精品一区二区三区蜜桃久 | www.788.com色淫免费 | 欧美国产日韩在线观看成人 | 91免费在线视频 | 久草视频网站 | 国产乱码久久久久久 | 天天干天天操天天爽 | 国产精品久久久久久久久久免费看 | 国产一区在线看 | 国产中文字幕在线观看 | 日韩福利在线 | aaa精品 | 久久www免费人成看片高清 | 国产精品久久久久久久久久久免费看 | 欧美中文字幕一区二区三区亚洲 | 久久久tv | 五月天国产| 国产精品免费大片 | 一区二区三区视频在线观看 | 久久久精品一区 | 男女在线免费观看 | 国产午夜精品久久久 | 91在线看网站 | 色网站视频 | 欧美一级大片免费观看 | 日韩一区二区三区四区五区 | 日日操视频 | 亚洲一区二区免费 | 欧美性视频在线播放 |