注解 实现原理 详解

Java 注解实现原理详解

注解(Annotation)是 Java 提供的一种元数据机制,用于为代码元素(类、方法、字段、参数等)添加额外的信息。注解不会直接影响程序逻辑,但可以通过 工具(如编译器、运行时框架) 解析和处理,完成特定的功能。


1. 注解的分类

1.1 按 生命周期 分类:
  1. 编译期注解(SOURCE):

    • 只存在于源码中,编译后会被丢弃。
    • 用于代码检查、生成文档、自动生成代码。
    • 示例:@Override, @Deprecated, @SuppressWarnings
  2. 类加载期注解(CLASS):

    • 存在于编译后的字节码文件中,但 JVM 加载类时不会保留。
    • 常用于字节码增强工具,如 Lombok。
  3. 运行期注解(RUNTIME):

    • JVM 加载类后仍然保留,可以通过反射获取。
    • 常用于运行时动态功能(如 Spring、MyBatis)。
1.2 按功能分类:
  1. 标记注解

    • 没有属性,起标记作用。
    • 示例:@Override, @FunctionalInterface
  2. 单值注解

    • 只有一个属性,属性名通常为 value
    • 示例:@SuppressWarnings("unchecked")
  3. 完整注解

    • 包含多个属性,支持默认值。
    • 示例:
      @Retention(RetentionPolicy.RUNTIME)
      @Target(ElementType.METHOD)
      public @interface MyAnnotation {String name();int age() default 18;
      }
      

2. 注解的实现原理

2.1 注解的本质
  1. 注解是接口

    • 编译时,注解被编译为实现了 java.lang.annotation.Annotation 接口的特殊类。
    • 注解中的属性实际上是接口的方法。
    • 示例:
      @interface MyAnnotation {String value();
      }
      
      等价于:
      public interface MyAnnotation extends java.lang.annotation.Annotation {String value();
      }
      
  2. 注解的元信息存储在字节码中

    • 编译器在编译注解时,会将元数据存储到 .class 文件的 属性表 中,根据注解的生命周期决定是否保留到运行时。
2.2 注解的处理方式
  1. 编译期处理(APT)

    • 使用注解处理器(javax.annotation.processing 包)分析和处理注解。
    • 注解处理器会在编译期扫描 SOURCE 注解,并生成额外的代码或文件。

    示例:

    @SupportedAnnotationTypes("com.example.MyAnnotation")
    @SupportedSourceVersion(SourceVersion.RELEASE_8)
    public class MyAnnotationProcessor extends AbstractProcessor {@Overridepublic boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {for (Element element : roundEnv.getElementsAnnotatedWith(MyAnnotation.class)) {System.out.println("Processing: " + element);}return true;}
    }
    
  2. 运行期处理(反射)

    • 通过 Java 反射 API 获取运行时注解,并动态执行逻辑。
    • 示例:
      @Retention(RetentionPolicy.RUNTIME)
      @Target(ElementType.METHOD)
      public @interface MyAnnotation {String value();
      }public class Test {@MyAnnotation("Hello")public void myMethod() {}
      }public static void main(String[] args) throws Exception {Method method = Test.class.getMethod("myMethod");if (method.isAnnotationPresent(MyAnnotation.class)) {MyAnnotation annotation = method.getAnnotation(MyAnnotation.class);System.out.println(annotation.value()); // 输出 "Hello"}
      }
      

3. 注解的组成与元注解

3.1 元注解(Meta-Annotation)

元注解是用于定义注解行为的注解。Java 提供了 5 种常见的元注解:

元注解功能
@Retention指定注解的生命周期(SOURCE, CLASS, RUNTIME)。
@Target指定注解可以应用的代码元素(类、方法、字段等)。
@Documented表示注解会被 Javadoc 工具处理,生成的文档包含注解信息。
@Inherited表示注解可以被子类继承。
@Repeatable允许同一个注解在同一位置多次使用。
3.2 示例:自定义注解
import java.lang.annotation.*;@Retention(RetentionPolicy.RUNTIME)  // 运行时保留
@Target({ElementType.METHOD, ElementType.TYPE})  // 可用于类和方法
@Documented  // 生成文档时包含
@Inherited  // 可被子类继承
public @interface MyAnnotation {String value();  // 属性int age() default 18;  // 属性,具有默认值
}

使用:

@MyAnnotation(value = "Test", age = 25)
public class MyClass {@MyAnnotation("Method Annotation")public void myMethod() {}
}

4. 注解在框架中的应用

4.1 Spring 中的注解
  1. 声明式注解

    • @Component@Service:标记类为 Spring 容器的组件。
    • @Autowired:自动注入 Bean。
    • @Transactional:声明事务管理。
  2. AOP 和运行期处理

    • Spring AOP 使用运行期注解(如 @Transactional)配合动态代理,实现方法拦截。
4.2 Hibernate 中的注解
  • @Entity:标注类为数据库实体。
  • @Table:映射数据库表。
  • @Column:映射字段。
  • 使用运行期注解解析实现对象关系映射(ORM)。

5. 注解的实现细节与性能

5.1 字节码中的存储形式

注解的元信息存储在 .class 文件的 属性表RuntimeVisibleAnnotationsRuntimeInvisibleAnnotations)中,通过反射或字节码工具读取。

5.2 注解的性能
  • 编译期注解无运行时开销。
  • 运行期注解有反射开销,但对于少量操作性能影响可忽略。
  • 过多的运行期注解解析可能影响应用启动时间。

6. 注解的进阶使用

6.1 注解组合

将多个注解组合在一起,简化使用。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Component
@Transactional
public @interface ServiceTransactional {
}
6.2 注解处理框架
  1. Spring
    • 利用注解实现依赖注入、AOP 和事务管理。
  2. Lombok
    • 编译期生成代码,如 @Getter, @Setter
  3. MyBatis
    • 使用 @Mapper 映射 SQL 语句。

7. 总结

注解是 Java 提供的元编程工具,通过元注解定义注解的行为,通过编译期处理器或运行期反射实现注解的功能。在现代框架(如 Spring、Hibernate、MyBatis)中,注解简化了配置和开发流程,是开发高效、简洁代码的核心工具。理解注解的实现原理、生命周期以及应用场景是掌握 Java 注解编程的关键。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/63044.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用Feign远程调用丢失请求头问题

在使用Feign进行远程调用时&#xff0c;当前服务是能拿到请求头信息的&#xff0c;请求头包含有登录认证Cookie等重要信息&#xff0c;但是在调用远程服务时&#xff0c;远程服务却拿不到请求头信息&#xff0c;因为使用Feign进行远程调用实际上是发起新的Request请求了&#x…

SpringBoot 整合 Avro 与 Kafka 详解

SpringBoot 整合 Avro 与 Kafka 详解 在大数据处理和实时数据流场景中&#xff0c;Apache Kafka 和 Apache Avro 是两个非常重要的工具。Kafka 作为一个分布式流处理平台&#xff0c;能够高效地处理大量数据&#xff0c;而 Avro 则是一个用于序列化数据的紧凑、快速的二进制数…

2021数学分析【南昌大学】

2021 数学分析 求极限 lim ⁡ n → ∞ 1 n ( n + 1 ) ( n + 2 ) ⋯ ( n + n ) n \lim_{n \to \infty} \frac{1}{n} \sqrt [n]{(n+1)(n+2) \cdots (n+n)} n→∞lim​n1​n(n+1)(n+2)⋯(n+n) ​ lim ⁡ n → ∞ 1 n ( n + 1 ) ( n + 2 ) ⋯ ( n + n ) n = lim ⁡ n → ∞ ( n + …

vue+mars3d点击图层展示炫酷的popup弹窗

展示效果 目录 一&#xff1a;叠加数据图层到地图上&#xff0c;此时需要使用bindPopup绑定popup 二、封装自定义的popup&#xff0c;样式可以自行调整 一&#xff1a;叠加数据图层到地图上&#xff0c;此时需要使用bindPopup绑定popup 这里我根据数据不同&#xff0c;展示的…

【Python】用Python和Paramiko实现远程服务器自动化管理

《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 在现代IT环境中,远程服务器管理已成为运维工作的常态。随着自动化运维的需求不断增加,如何高效地管理远程服务器,提升操作的灵活性和效率…

框架建设实战2——创建frame-bom和out-bom

根frame-parent的创建一样&#xff0c;分别创建两个工程。主要管理公司内部的二方包和外部的三方包。 1.frame-bom GAV定义为&#xff1a; <groupId>com.test</groupId> <artifactId>frame-bom</artifactId> <name>frame-bom</name> …

【AIGC】如何使用高价值提示词Prompt提升ChatGPT响应质量

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: AIGC | 提示词Prompt应用实例 文章目录 &#x1f4af;前言&#x1f4af;提示词英文模板&#x1f4af;提示词中文解析1. 明确需求2. 建议额外角色3. 角色确认与修改4. 逐步完善提示5. 确定参考资料6. 生成和优化提示7.…

FPGA存在的意义:为什么adc连续采样需要fpga来做,而不会直接用iic来实现

FPGA存在的意义&#xff1a;为什么adc连续采样需要fpga来做&#xff0c;而不会直接用iic来实现 原因ADS111x连续采样实现连续采样功能说明iic读取adc的数据速率 VS adc连续采样的速率adc连续采样的速率iic读取adc的数据速率结论分析 FPGA读取adc数据问题一&#xff1a;读取adc数…

千益畅行,旅游卡有些什么优势?

千益畅行共享旅游卡是一种创新的旅游服务模式&#xff0c;旨在通过整合各类旅游资源&#xff0c;为用户提供一站式的旅游解决方案。这张旅游卡支持2至6人同行&#xff0c;涵盖了接机、酒店、用餐、大巴、导游、景区门票等服务&#xff0c;用户只需自行承担往返交通费用即可享受…

LobeChat-46.6k星!顶级AI工具集,一键部署,界面美观易用,ApiSmart 是你肉身体验学习LLM 最好IDEA 工具

LobeChat LobeChat的开源&#xff0c;把AI功能集合到一起&#xff0c;真的太爽了。 我第一次发现LobeChat的时候&#xff0c;就是看到那炫酷的页面&#xff0c;这么强的前端真的是在秀肌肉啊&#xff01; 看下它的官网&#xff0c;整个网站的动效简直闪瞎我&#xff01; GitH…

[报错] Error: PostCSS plugin autoprefixer requires PostCSS 8 问题解决办法

报错&#xff1a;Error: PostCSS plugin autoprefixer requires PostCSS 8 原因&#xff1a;autoprefixer版本过高 解决方案&#xff1a; 降低autoprefixer版本 执行&#xff1a;npm i postcss-loader autoprefixer8.0.0 参考&#xff1a; Error: PostCSS plugin autoprefix…

基于STM32的Wi-Fi无人机项目

引言 随着无人机技术的快速发展&#xff0c;基于微控制器的DIY无人机变得越来越流行。本项目将介绍如何使用STM32微控制器制作一架简单的Wi-Fi无人机。通过本项目&#xff0c;您将了解到无人机的基本组成部分&#xff0c;如何进行硬件连接&#xff0c;代码编写&#xff0c;以及…

IDEA注释格式、匹配补全调整

1.注释格式调整 目前重新捡起一部分Java&#xff0c;写代码时候发现注释快捷键总是放在第一列&#xff0c;看起来很难受&#xff0c;故寻找方法如下&#xff1a; 分别点击 编辑器-代码样式-Java 修改注释代码选项如下 2.大小写匹配补全问题 还发现在写代码过程中&#xff0c…

麒麟 V10(ky10.x86_64)无网环境下 openssl - 3.2.2 与 openssh - 9.8p1 升级【最全教程】

目录 背景 安装包下载 上传解压安装包 安装zlib 安装OpenSSL 安装OpenSSH 验证 背景 近期&#xff0c;项目上线已进入倒计时阶段&#xff0c;然而在至关重要的安全检查环节中&#xff0c;却惊现现有的 OpenSSH 存在一系列令人担忧的漏洞&#xff1a; OpenSSH 资源管理错…

Android EventBus最全面试题及参考答案

目录 什么是 EventBus? 请解释 EventBus 是什么,以及它的工作原理。 简述 EventBus 的工作原理。 EventBus 的主要组成部分有哪些? EventBus 是如何实现发布订阅模式的? EventBus 与观察者模式有什么区别? EventBus 的优点在哪、不用 EventBus 怎么解决? EventBu…

高级架构二 Git基础到高级

一 Git仓库的基本概念和流程 什么是版本库&#xff1f;版本库又名仓库&#xff0c;英文名repository,你可以简单的理解一个目录&#xff0c;这个目录里面的所有文件都可以被Git管理起来&#xff0c;每个文件的修改&#xff0c;删除&#xff0c;Git都能跟踪&#xff0c;以便任何…

从excel数据导入到sqlsever遇到的问题

1、格式问题时间格式&#xff0c;excel中将日期列改为日期未生效&#xff0c;改完后&#xff0c;必须手动单击这个单元格才能生效&#xff0c;那不可能一个一个去双击。解决方案如下 2、导入之后表字段格式问题&#xff0c;数据类型的用navicat导入之后默认是nvarchar类型的&a…

21天掌握javaweb--->第4天:MyBatis-Plus基础与进阶

21天掌握JavaWeb--->第4天&#xff1a;MyBatis-Plus基础与进阶 MyBatis-Plus简介 MyBatis-Plus&#xff08;简称MP&#xff09;是一个MyBatis的增强工具&#xff0c;在MyBatis的基础上只做增强不做改变&#xff0c;为简化开发、提高效率而生。它具有以下核心优势&#xff…

FREERTOS二值信号量实验

代码&#xff1a; 主程序 #include "./SYSTEM/sys/sys.h" #include "./SYSTEM/usart/usart.h" #include "./SYSTEM/delay/delay.h" #include "./BSP/LED/led.h" #include "./BSP/LCD/lcd.h" #include "./BSP/KEY/key…

vue3中 axios 发送请求 刷新token 封装axios

service.js 页面 import axios from axios // 创建axios实例 const instance axios.create({baseURL: http://gcm-test.jhzhkj.cn:8600/h5card/,timeout: 5000, // 请求超时时间headers: {get: {Content-Type: application/x-www-form-urlencoded},post: {Content-Type: appl…