0

0

Java中的元注解(@Target, @Retention)详解_定义注解的有效范围

P粉602998670

P粉602998670

发布时间:2026-02-18 12:43:03

|

451人浏览过

|

来源于php中文网

原创

@target 和 @retention 共同决定注解的使用位置与存活周期:前者校验语法合法性,后者控制反射可读性;二者独立生效,缺一不可,且需匹配实际处理逻辑。

java中的元注解(@target, @retention)详解_定义注解的有效范围

@Target 指定注解能用在哪儿,写错就直接编译失败

Java 编译器会严格校验 @Target 声明的元素类型是否匹配实际使用位置。比如你定义了一个只允许标注方法的注解,却把它加在类上,javac 会立刻报错:java.lang.annotation.AnnotationTypeMismatchException 或更常见的 Cannot apply @XXX to class

常见错误是以为 ElementType.TYPE 能覆盖内部类、枚举、接口——它确实可以,但如果你还希望注解能用在字段上,就必须显式加上 ElementType.FIELD,不能靠“继承”或“默认包含”。

  • ElementType.METHOD:仅方法,不包括构造器(构造器要用 CONSTRUCTOR
  • ElementType.PARAMETER:仅普通方法/构造器参数,Lambda 表达式参数不算
  • ElementType.TYPE_USE:这是个“补丁型”类型,用于泛型、异常声明、new 表达式等上下文,和 TYPE 并不互斥,常需同时声明
  • 多个值用大括号:@Target({ElementType.METHOD, ElementType.TYPE_USE})

@Retention 控制注解存活时间,影响运行时能否反射读取

如果想在运行时通过 getAnnotations() 拿到自定义注解,@Retention 必须设为 RetentionPolicy.RUNTIME。设成 CLASS(默认值)的话,注解保留在字节码里但 JVM 不加载;设成 SOURCE 则只存在于源码,编译完就丢弃。

容易被忽略的是:Spring 的 @Transactional、Lombok 的 @Data 等都依赖 RUNTIME 级别才能工作。而像 @Override 这种只给编译器看的,就是 SOURCE 级别。

立即学习Java免费学习笔记(深入)”;

亿众购物系统
亿众购物系统

一套设计完善、高效的web商城解决方案,独有SQL注入防范、对非法操作者锁定IP及记录功能,完整详细的记录了非法操作情况,管理员可以随时查看网站安全日志以及解除系统自动锁定的IP等前台简介:  1)系统为会员制购物,无限会员级别。  2)会员自动升级、相应级别所享有的折扣不同。  3)产品可在缺货时自动隐藏。  4)自动统计所有分类中商品数量,并在商品分类后面显示。  5)邮件列表功能,可在线订阅

下载
  • 反射读取注解前,先确认类加载后该注解是否还在:MyAnnotation.class.getAnnotation(MyAnnotation.class) != null
  • Android 开发中注意:ProGuard/R8 可能移除 RUNTIME 注解,需保留规则:-keepattributes RuntimeVisibleAnnotations
  • RetentionPolicy.CLASS 在某些 AOP 场景下够用,且比 RUNTIME 略轻量,但多数框架要求明确写 RUNTIME

组合使用时,@Target 和 @Retention 的约束是独立的

这两个元注解之间没有隐含依赖关系。你可以写一个 @Retention(RetentionPolicy.SOURCE)@Target(ElementType.ANNOTATION_TYPE) 的注解——意思是“这个注解只能用来标注其他注解,且只在编译期存在”。这在定义元注解时很常见(比如 Spring 的 @Documented 就是 SOURCE + ANNOTATION_TYPE)。

但反过来说,如果误把 @Retention(RetentionPolicy.SOURCE) 用在需要运行时处理的注解上,反射永远拿不到它,调试时会卡在“注解明明写了,为什么 null?”

  • 检查路径:注解定义 → @Retention 值 → 实际使用处是否在 @Target 允许范围内 → 运行时是否调用了正确的反射 API(如 getMethod().getAnnotation() 而非 getClass().getAnnotation()
  • IDE 通常不会高亮 @Retention 错误,只有编译或运行时才暴露问题
  • 不要为了“保险”把 @Target 设成 ElementType.ANY,它会让注解滥用变得难以追踪

自定义注解生效的前提:得让处理逻辑真正触发

光有 @Target@Retention 不等于注解就有用。它们只是“挂载许可”和“存活许可”,真正干活的是你写的处理器——可能是编译期注解处理器(javax.annotation.processing.Processor),也可能是运行时通过反射+条件判断的手动逻辑。

比如你写了 @LogExecutionTime 并设了 @Retention(RUNTIME),但没配 Spring AOP 切面或没写 MethodInterceptor,那它就是个摆设。

  • 编译期处理:需在 META-INF/services/javax.annotation.processing.Processor 中声明处理器类名,并确保 javac -processorpath 包含对应 jar
  • 运行时处理:必须主动调用 getDeclaredAnnotations() 或框架(如 Spring、Micrometer)已注册的扫描逻辑
  • 注解本身不带逻辑,所有行为都来自外部代码对它的响应——这点初学者最容易误解
事情说清了就结束。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

142

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

81

2026.01.26

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

244

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

766

2024.03.01

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

58

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1508

2023.10.19

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

相关下载

更多

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.7万人学习

C# 教程
C# 教程

共94课时 | 9.7万人学习

Java 教程
Java 教程

共578课时 | 67.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号