欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中注解的常見用法總結(jié)

 更新時間:2023年07月21日 08:33:28   作者:Java技術(shù)指北  
注解(Annotation),也叫元數(shù)據(jù),是JDK1.5及以后版本引入的一個特性,本文主要為大家介紹了注解的常見用法,需要的小伙伴可以參考一下

Annotation

注解(Annotation),也叫元數(shù)據(jù)。一種代碼級別的說明。它是JDK1.5及以后版本引入的一個特性,與類、接口、枚舉是在同一個層次。它可以聲明在包、類、字段、方法、局部變量、方法參數(shù)等的前面,用來對這些元素進(jìn)行說明,注釋。作用分類:

  • 編寫文檔:通過代碼里標(biāo)識的元數(shù)據(jù)生成文檔【生成文檔doc文檔】
  • 代碼分析:通過代碼里標(biāo)識的元數(shù)據(jù)對代碼進(jìn)行分析【使用反射】
  • 編譯檢查:通過代碼里標(biāo)識的元數(shù)據(jù)讓編譯器能夠?qū)崿F(xiàn)基本的編譯檢查【Override】

注解不會改變程序的語義,只是作為注解(標(biāo)識)存在,我們可以通過反射機(jī)制編程實現(xiàn)對這些元數(shù)據(jù)(用來描述數(shù)據(jù)的數(shù)據(jù))的訪問

分類

  • 運行期注解 程序運行時才會被解析到的注解,一般通過反射機(jī)制來實現(xiàn),很多框架中都會用到,經(jīng)常會看到一個注解和一些簡單的配置來實現(xiàn)非常復(fù)雜的功能
  • 編譯期注解 一般用來解析類型元數(shù)據(jù),根據(jù)特定注解解析并生成代碼,或者生成一些描述性文件,比如properties、json等,比如為Pojo生成getter和setter方法

關(guān)鍵注解

@java.lang.annotation.Retention定義注解的有效時期

相關(guān)參數(shù):RetentionPolicy.SOURCE: 編譯期生效,編譯器會丟棄,編譯后的class文件并不包含該注解 RetentionPolicy.CLASS:  注解會被保留在class文件中,但是運行期不會生效,被JVM忽略 RetentionPolicy.RUNTIME: 注解會被保留在class文件中,并且會在運行期生效,JVM會讀取

@Target定義注解作用對象,也就是注解是可以用在類、方法、參數(shù)還是其他等待

相關(guān)參數(shù):ElementType.TYPE: 該注解只能運用到Class, Interface, enum上 ElementType.FIELD: 該注解只能運用到Field上 ElementType.METHOD: 該注解只能運用到方法上 ElementType.PARAMETER: 該注解只能作用在參數(shù)上 ElementType.CONSTRUCTOR: 該注解只能作用在構(gòu)造方法上 ElementType.LOCAL_VARIABLE: 該注解作用在本地變量或catch語句 ElementType.ANNOTATION_TYPE: 該注解只能作用在注解上 ElementType.PACKAGE: 該注解只能用在包上

Java中常見的內(nèi)置注解:

  • @Override
  • @Deprecated
  • @SuppressWarnings

繼承關(guān)系

1.@Inherited

如果某個注解上有@Inherited注解,當(dāng)查找該類型的注解時,會先查找目標(biāo)類型是否存在注解,如果有,直接返回;否則,繼續(xù)在父類上尋找注解, 停止的條件為在父類上找到該類型的注解或者父類為Object類型。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Inherited
public?@interface?ClassMapper?{
}

下面的示例中,如果ClassMapper沒有@Inherited修飾,則返回null

Child.class.getAnnotation(ClassMapper.class);
@Slf4j
public?class?ExtendAnnotationTests?{
????@ClassMapper
????public?class?Demo?{?}
????public?class?Child?extends?Demo{??}
}

2.元注解(注解上的注解)

我們知道,在Spring中,注解@Service與@Component都是用來標(biāo)記類,交由Spring容器管理其對應(yīng)的Bean,是結(jié)果是等效的。主要是Spring將注解和元注解進(jìn)行了合并

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public?@interface?Mapper?{
}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Mapper
public?@interface?ClassMapper?{
}

通過下面的方法可以拿到元注解,從而進(jìn)行其他擴(kuò)展。

public?class?Tests?{
????@Test
????public?void?test(){
????????ClassMapper?classMapper?=?Demo.class.getAnnotation(ClassMapper.class);
????????log.info("classMapper:?{}",?classMapper);
????????Mapper?mapper?=?classMapper.annotationType().getAnnotation(Mapper.class);
????????log.info("mapper:?{}",?mapper);
????}
}

示例

示例主要針對@java.lang.annotation.Retention參數(shù)的三種情況,了解注解的生效時期:

RetentionPolicy.RUNTIME

該示例實現(xiàn)通過自定義注解@SystemProperty,實現(xiàn)為對象字段設(shè)置系統(tǒng)屬性

1.定義注解@SystemProperty

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
@Documented
public?@interface?SystemProperty?{
????String?value();
}

2.定義對象工廠

主要作用是在運行時解析注解@SystemProperty,并實現(xiàn)系統(tǒng)屬性注入的邏輯。前面說到,注解的作用主要是標(biāo)記,針對RetentionPolicy.RUNTIME類型的注解,一般是在運行時通過反射實現(xiàn)對注解標(biāo)識的類、字段或方法等元素處理的過程。

ObjectFactory是一個對象生產(chǎn)工廠,這樣我們可以在運行期解析目標(biāo)對象中的是否有@SystemProperty標(biāo)識的字段,并對該字段進(jìn)行值的設(shè)定,這是該注解設(shè)計的目的,但是具體實現(xiàn)需要我們根據(jù)需求來完成

@Slf4j
public?class?ObjectFactory?{
????//?省略?...
????public?static?<T>?T?getObject(Class<T>?type,?Object...?args){
????????Constructor<T>?constructor?=?findTypeConstructor(type,?args);
????????T?object?=?constructor.newInstance(args);
????????//?通過反射找到對象中@SystemProperty的字段,并根據(jù)其設(shè)置參數(shù)將系統(tǒng)屬性設(shè)定到該對象字段中
????????processFieldAnnotations(object,?type,?SystemProperty.class);
????????return?object;
????}
????//?省略?...??
}

3.驗證

可以查看對象中被注解標(biāo)識的屬性被設(shè)置上去了

@Slf4j
public?class?RuntimeAnnotationTests?{
????@Test
????public?void?run(){
????????Demo?demo?=?ObjectFactory.getObject(Demo.class);
????????log.info(">>?result:?{}",?demo.user);
????}
????@Data
????public?static?class?Demo{
????????@SystemProperty("user.name")
????????private?String?user;
????}
}

RetentionPolicy.CLASS

該示例主要實現(xiàn),編譯器判斷通過@FinalClass注解標(biāo)記的類是否為final類型

1.定義注解

@Retention(RetentionPolicy.CLASS)
@Target(ElementType.TYPE)
@Documented
public?@interface?FinalClass?{
}

2.編寫AbstractProcessor的實現(xiàn)

@SupportedAnnotationTypes({FinalClassProcessor.FINAL_CLASS})
@SupportedSourceVersion(SourceVersion.RELEASE_8)
@AutoService(Processor.class)
public?class?FinalClassProcessor?extends?AbstractProcessor?{
????public?static?final?String?FINAL_CLASS?=?"com.sucl.blog.jdk.annotation.compile.FinalClass";
????@Override
????public?boolean?process(Set<??extends?TypeElement>?annotations,?RoundEnvironment?roundEnv)?{
????????TypeElement?annotationType?=?this.processingEnv.getElementUtils().getTypeElement(FINAL_CLASS);
????????if(?annotationType?!=?null?){
????????????for?(Element?element?:?roundEnv.getElementsAnnotatedWith(annotationType))?{
????????????????if(?element?instanceof?TypeElement?){
????????????????????TypeElement?typeElement?=?(TypeElement)?element;
????????????????????if(?!typeElement.getModifiers().contains(Modifier.FINAL)?){
????????????????????????String?message?=?String.format("類【%s】必須為final類型",?typeElement);
????????????????????????this.processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,?message);
????????????????????}
????????????????}
????????????}
????????}
????????return?true;
????}
}

3.使FinalClassProcessor生效

基于google auto-service

3.1 添加依賴

????<dependency>
??????<groupId>com.google.auto.service</groupId>
??????<artifactId>auto-service</artifactId>
??????<version>1.1.0</version>
????</dependency>

3.2 在Processor通過注解@AutoService標(biāo)識

@AutoService(Processor.class)
public?class?FinalClassProcessor?extends?AbstractProcessor{}

基于maven插件

<plugin>
????<groupId>org.apache.maven.plugins</groupId>
????<artifactId>maven-compiler-plugin</artifactId>
????<configuration>
????????<annotationProcessors>
????????????<annotationProcessor>
????????????????com.sucl.blog.jdk.annotation.compile.FinalClassProcessor
????????????</annotationProcessor>
????????</annotationProcessors>
????</configuration>
</plugin>

4.驗證

打包,在項目中引入該jar,定義一個類,類似下面這樣,當(dāng)該類沒有final修飾時,通過maven install命令,可以看到控制臺打印自定義的錯誤信息

@FinalClass
public?final?class?ProcessorFinder?{}

注意

RetentionPolicy.CLASS的使用需要達(dá)打成jar包才行,不然無法再編譯時處理注解

RetentionPolicy.SOURCE

定義一個注解,通過打包后的結(jié)果觀察該注解的狀態(tài)

1.定義注解

@Retention(RetentionPolicy.SOURCE)
@Target(ElementType.TYPE)
@Documented
public?@interface?System?{
}

2.定義測試類,并通過@System修飾

@System
public?class?SystemProvider?{
}

3.打包,借助maven-source-plugin同時將源碼打包

????<plugins>
????????<plugin>
????????????<groupId>org.apache.maven.plugins</groupId>
????????????<artifactId>maven-source-plugin</artifactId>
????????????<version>3.2.1</version>
????????????<executions>
????????????????<execution>
????????????????????<id>attach-sources</id>
????????????????????<goals>
????????????????????????<goal>jar</goal>
????????????????????</goals>
????????????????</execution>
????????????</executions>
????????</plugin>
????</plugins>

4.在源碼包中,可以看到該注解仍然存在,但是class文件中卻沒有

在基于Spring Boot開發(fā)項目時,我們一般通過 @ConfigurationProperties 配合 spring-boot-configuration-processor,可以實現(xiàn)在項目打包時 生成一個spring-configuration-metadata.json的配置描述文件,這樣在編寫application.yml配置時,就會得到配置提示,其實現(xiàn)方式就是基于 ConfigurationMetadataAnnotationProcessor,

結(jié)束語

注解本身沒有含義,主要作用是標(biāo)記目標(biāo)元素,后續(xù)拿到改標(biāo)識的元數(shù)據(jù),進(jìn)行一系列的處理。注解的使用是非常廣泛的,各種框架中都使用頻繁,基于注解可以將很多抽象功能提取出來,通過簡單 的標(biāo)識來實現(xiàn)各種復(fù)雜的功能

以上就是Java中注解的常見用法總結(jié)的詳細(xì)內(nèi)容,更多關(guān)于Java注解用法的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • fasterxml jackson反序列化時對于非靜態(tài)內(nèi)部類報錯問題及解決

    fasterxml jackson反序列化時對于非靜態(tài)內(nèi)部類報錯問題及解決

    這篇文章主要介紹了fasterxml jackson反序列化時對于非靜態(tài)內(nèi)部類報錯問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • 員工管理系統(tǒng)java版

    員工管理系統(tǒng)java版

    這篇文章主要為大家詳細(xì)介紹了java版的員工管理系統(tǒng),,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • SpringBoot自定義注解使用讀寫分離Mysql數(shù)據(jù)庫的實例教程

    SpringBoot自定義注解使用讀寫分離Mysql數(shù)據(jù)庫的實例教程

    這篇文章主要給大家介紹了關(guān)于SpringBoot自定義注解使用讀寫分離Mysql數(shù)據(jù)庫的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • Java中StringBuilder類的用法解析

    Java中StringBuilder類的用法解析

    StringBuilder是一個可變的字符序列,這個類提供了一個與StringBuffer兼容的API。本文主要為大家介紹了StringBuilder類的常用用法,需要的可以參考一下
    2023-05-05
  • SpringBoot AOP方式實現(xiàn)多數(shù)據(jù)源切換的方法

    SpringBoot AOP方式實現(xiàn)多數(shù)據(jù)源切換的方法

    本篇文章主要介紹了SpringBoot AOP方式實現(xiàn)多數(shù)據(jù)源切換的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • 詳解Spring的StringUtils踩坑記錄

    詳解Spring的StringUtils踩坑記錄

    這篇文章主要介紹了詳解Spring的StringUtils踩坑記錄,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • Flowable整合SpringBoot實現(xiàn)的示例代碼

    Flowable整合SpringBoot實現(xiàn)的示例代碼

    本文詳細(xì)介紹了如何在SpringBoot項目中整合Flowable進(jìn)行工作流管理,包括依賴引入、流程部署與啟動、表結(jié)構(gòu)、流程掛起和激活以及任務(wù)分配等關(guān)鍵操作,具有一定的參考價值,感興趣的可以了解一下
    2024-09-09
  • Spring常用數(shù)據(jù)源的xml配置詳解

    Spring常用數(shù)據(jù)源的xml配置詳解

    這篇文章主要介紹了Spring常用數(shù)據(jù)源的xml配置詳解,數(shù)據(jù)源是連接到數(shù)據(jù)庫的一類路徑,它包含了訪問數(shù)據(jù)庫的信息(地址、用戶名、密碼),數(shù)據(jù)源就像是排水管道,需要的朋友可以參考下
    2023-07-07
  • SpringMVC KindEditor在線編輯器之文件上傳代碼實例

    SpringMVC KindEditor在線編輯器之文件上傳代碼實例

    這篇文章主要介紹了SpringMVC KindEditor在線編輯器之文件上傳代碼實例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-09-09
  • springboot?vue前后端接口測試樹結(jié)點添加功能

    springboot?vue前后端接口測試樹結(jié)點添加功能

    這篇文章主要為大家介紹了springboot?vue前后端接口測試樹結(jié)點添加功能,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-05-05

最新評論