0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Java中常見的注解

科技綠洲 ? 來源:Java技術指北 ? 作者:Java技術指北 ? 2023-09-30 15:35 ? 次閱讀

Annotation

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

  1. 編寫文檔:通過代碼里標識的元數據生成文檔【生成文檔doc文檔】
  2. 代碼分析:通過代碼里標識的元數據對代碼進行分析【使用反射】
  3. 編譯檢查:通過代碼里標識的元數據讓編譯器能夠實現基本的編譯檢查【Override】

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

分類

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

關鍵注解

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

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

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

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


Java中常見的內置注解:

  • @Override
  • @Deprecated
  • @SuppressWarnings

繼承關系

  • @Inherited

如果某個注解上有@Inherited注解,當查找該類型的注解時,會先查找目標類型是否存在注解,如果有,直接返回;否則,繼續(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{  }
}
  • 元注解 (注解上的注解)

我們知道,在Spring中,注解@Service與@Component都是用來標記類,交由Spring容器管理其對應的Bean,是結果是等效的。主要是Spring將注解和元注解進行了合并

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Mapper {

}

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

}

通過下面的方法可以拿到元注解,從而進行其他擴展。

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參數的三種情況,了解注解的生效時期:

RetentionPolicy.RUNTIME

該示例實現通過自定義注解@SystemProperty,實現為對象字段設置系統屬性

  1. 定義注解@SystemProperty
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
@Documented
public @interface SystemProperty {

    String value();
}
  1. 定義對象工廠

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

ObjectFactory是一個對象生產工廠,這樣我們可以在運行期解析目標對象中的是否有@SystemProperty標識的字段,并對該字段進行值的設定,這是該注解設計的目的,但是具體實現需要我們根據需求來完成

@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的字段,并根據其設置參數將系統屬性設定到該對象字段中
        processFieldAnnotations(object, type, SystemProperty.class);
        return object;
    }
    
    // 省略 ...  
}
  1. 驗證
    可以查看對象中被注解標識的屬性被設置上去了
@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

該示例主要實現,編譯器判斷通過@FinalClass注解標記的類是否為final類型

  1. 定義注解
@Retention(RetentionPolicy.CLASS)
@Target(ElementType.TYPE)
@Documented
public @interface FinalClass {

}
  1. 編寫AbstractProcessor的實現
@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;
    }
}
  1. 使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標識

@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 >
  1. 驗證

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

@FinalClass
public final class ProcessorFinder {}

圖片

注意

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

RetentionPolicy.SOURCE

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

  1. 定義注解
@Retention(RetentionPolicy.SOURCE)
@Target(ElementType.TYPE)
@Documented
public @interface System {
    
}
  1. 定義測試類,并通過@System修飾
@System
public class SystemProvider {

}
  1. 打包,借助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 >
  1. 在源碼包中,可以看到該注解仍然存在,但是class文件中卻沒有

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

結束語

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

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯系本站處理。 舉報投訴
  • 接口
    +關注

    關注

    33

    文章

    8447

    瀏覽量

    150722
  • JAVA
    +關注

    關注

    19

    文章

    2952

    瀏覽量

    104479
  • 代碼
    +關注

    關注

    30

    文章

    4722

    瀏覽量

    68231
  • 編譯器
    +關注

    關注

    1

    文章

    1617

    瀏覽量

    49015
  • 元數據
    +關注

    關注

    0

    文章

    32

    瀏覽量

    9119
收藏 人收藏

    評論

    相關推薦

    如何通過注解來優(yōu)化我們的Java代碼

    Java注解可以說是我們編碼過程中最常用的。本篇文章將給大家介紹Java注解的概念、作用以及如何使用注解來提升代碼的可讀性和靈活性,并介紹如
    的頭像 發(fā)表于 09-30 11:39 ?578次閱讀

    FPGACPLD中常見模塊設計精華集錦

    FPGACPLD中常見模塊設計精華集錦
    發(fā)表于 08-17 22:22

    HarmonyOS功能開發(fā)中常見問題

    功能開發(fā)中常見的問題如何查看“config.json”文件的所有字段說明?“config.json”文件的各字段說明請查閱配置文件的元素。怎么實現Ability可以被其他應用調用?開發(fā)者需要
    發(fā)表于 09-17 18:25

    電源中常見的故障現象如何維修

    電源中常見的故障現象如何維修
    發(fā)表于 03-11 07:24

    詳細介紹了Java泛型、注解、并發(fā)編程

    介紹了Java泛型、注解、并發(fā)編程、數據傳輸與序列化、高效IO、容器集合、反射與類加載以及JVM重點知識線程、內存模型、JVM運行時內存、垃圾回收與算法、Java中四種引用類型、GC 分代收集算法
    發(fā)表于 08-20 06:09

    HarmonyOS注解的使用方法分享

    定義我們的注解自定義注解1、聲明注解功能:檢測類中是否有規(guī)范的get方法新建java libray的module,命名為annotation,創(chuàng)建
    發(fā)表于 03-28 14:04

    分析java注解基本概念

    什么是注解(Annotation): Annotation(注解)就是Java提供了一種元程序中的元素關聯任何信息和著任何元數據(metadata)的途徑和方法。Annotion(注解
    發(fā)表于 09-27 14:53 ?0次下載

    Spring Boot中常見的各類型注解的使用方式

    大家好,我是程序汪,企業(yè)開發(fā)項目SpringBoot已經是必備框架了,其中注解是開發(fā)中的小工具(誰處可見哦),用好了開發(fā)效率大大提升,當然用錯了也會引入缺陷。
    的頭像 發(fā)表于 06-20 16:38 ?1731次閱讀

    注解定義Bean及開發(fā)

    注解本質是一個繼承了Annotation 的特殊接口,其具體實現類是Java 運行時生成的動態(tài)代理類。
    發(fā)表于 08-02 10:26 ?428次閱讀

    JAVA注解是怎么做到的(上)

    注解想必大家在項目中經常使用,比如Spring框架中常用的一些注解:`@Controller`、`@Service`、`@RequestMapping`等等,它是JDK1.5及以后版本引入的一個特性
    的頭像 發(fā)表于 05-11 10:57 ?613次閱讀

    JAVA注解是怎么做到的(下)

    注解想必大家在項目中經常使用,比如Spring框架中常用的一些注解:`@Controller`、`@Service`、`@RequestMapping`等等,它是JDK1.5及以后版本引入的一個特性
    的頭像 發(fā)表于 05-11 10:57 ?540次閱讀
    <b class='flag-5'>JAVA</b>中<b class='flag-5'>注解</b>是怎么做到的(下)

    3分鐘純Java注解搭個管理系統

    Erupt一個通用后臺管理框架,據說有 超低代碼量 、 零前端代碼 、零 CURD操作 、無需建表 ,純Java注解開發(fā) 等特色,號稱三分鐘就可以搭建一個完整的后臺管理系統。
    的頭像 發(fā)表于 07-28 11:27 ?1031次閱讀
    3分鐘純<b class='flag-5'>Java</b><b class='flag-5'>注解</b>搭個管理系統

    理解STM32控制中常見的PID算法

    理解STM32控制中常見的PID算法
    的頭像 發(fā)表于 10-17 17:28 ?2324次閱讀
    理解STM32控制<b class='flag-5'>中常見</b>的PID算法

    JDK中常見的Lamada表達式

    時事半功倍。 1 JDK中常見的Lamada表達式 Java中可以使用Lamada表達式的接口都有@FunctionalInterface注解。 先來看看util.function包下面含有FunctionalInterface
    的頭像 發(fā)表于 10-10 15:07 ?482次閱讀
    JDK<b class='flag-5'>中常見</b>的Lamada表達式

    springmvc常用5種注解

    SpringMVC是一種基于Java的Web框架,使用注解可以更加方便靈活地開發(fā)和管理控制器,實現請求的映射和處理。在SpringMVC中,有許多常用的注解,本文將詳細介紹其中的五種注解
    的頭像 發(fā)表于 11-22 16:51 ?766次閱讀