java 知识点:注解及使用
注解
- 大多数时候,我们会使用注解,而不是自定义注解。
- 注解给谁用?编译器 、给解析程序用
- 注解不是程序的一部分,可以理解为注解就是一个标签
主要的作用有以下四方面:
-
生成文档,通过代码里标识的元数据生成 javadoc 文档。
-
编译检查,通过代码里标识的元数据让编译器在编译期间进行检查验证。
-
编译时动态处理,编译时通过代码里标识的元数据动态处理,例如动态生成代码。
-
运行时动态处理,运行时通过代码里标识的元数据动态处理,例如使用反射注入实例。
注解的本质
package com.jj.anno;import java.lang.annotation.Retention;
import java.lang.annotation.Target;/*** 拥有宠物的信息* 宠物的名字* 宠物的年龄*/
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(java.lang.annotation.ElementType.TYPE)
public @interface Pet {String name();int age();
}
反编译后发现,注解本质上是一个 interface并且继承了 java.lang.annotation.Annotation这个接口.
➜ anno git:(master) ✗ javac Pet.java && javap Pet
警告: 二进制文件Pet包含com.jj.anno.Pet
Compiled from "Pet.java"
public interface com.jj.anno.Pet extends java.lang.annotation.Annotation {public abstract java.lang.String name();public abstract int age();
}
Java 注解(Annotation)是一种元数据形式,提供有关程序代码的额外信息,但这些信息不会影响代码的实际执行。注解可以用于类、方法、变量、参数等程序元素上,以便在编译时、类加载时或运行时进行处理。
元注解
元注解(Meta-Annotations)是用于注解其他注解的注解。元注解提供了对注解进行更精细的控制和定义。
@Retention
@Retention
用于指定注解的保留策略,即注解可以在何时可用。它可以被用于定义其他注解。
public enum RetentionPolicy {/*** Annotations are to be discarded by the compiler.*/SOURCE,/*** Annotations are to be recorded in the class file by the compiler* but need not be retained by the VM at run time. This is the default* behavior.*/CLASS,/*** Annotations are to be recorded in the class file by the compiler and* retained by the VM at run time, so they may be read reflectively.** @see java.lang.reflect.AnnotatedElement*/RUNTIME
}
@Retention
元注解有一个 value 属性,用于指定保留策略。可选的保留策略包括:
RetentionPolicy.SOURCE
:注解仅在源代码级别可见,在编译之后不会包含在编译后的字节码中。RetentionPolicy.CLASS
:注解在编译时保留,在编译后的字节码中可见,但在运行时不可通过 反射 获取注解信息(默认值)。RetentionPolicy.RUNTIME
:注解在运行时保留,可以通过反射获取注解信息,并根据注解的定义执行相应的代码,对运行时的代码有影响。
自定义注解的保留策略如果不指定,默认为 RetentionPolicy.CLASS
。
编译器分别使用了 RuntimeInvisibleAnnotations
和 RuntimeVisibleAnnotations
属性去记录了 RetentionPolicy.CLASS
注解的方法 和 RetentionPolicy.RUNTIME
方法的注解信息。
@Target
描述注解的作用位置
public enum ElementType {/** Class, interface (including annotation interface), enum, or record* declaration */TYPE,/** Field declaration (includes enum constants) */FIELD,/** Method declaration */METHOD,/** Formal parameter declaration */PARAMETER,/** Constructor declaration */CONSTRUCTOR,/** Local variable declaration */LOCAL_VARIABLE,/** Annotation interface declaration (Formerly known as an annotation type.) */ANNOTATION_TYPE,/** Package declaration */PACKAGE,/*** Type parameter declaration** @since 1.8*/TYPE_PARAMETER,/*** Use of a type** @since 1.8*/TYPE_USE,/*** Module declaration.** @since 9*/MODULE,/*** Record component** @jls 8.10.3 Record Members* @jls 9.7.4 Where Annotations May Appear** @since 16*/RECORD_COMPONENT;
}
@Documented
@Documented
:用于指定被它注解的注解是否应该包含在自动生成的 API 文档(Javadoc)中。它可以被用于定义其他注解。@Documented
:只是一个标记,它本身并不会影响注解的使用和行为。只有在生成 API 文档时才会体现其作用。
通常情况下,如果开发者希望在 API 文档中包含某个注解的信息,就可以为该注解添加 @Documented 元注解。如果不需要在 API 文档中包含注解信息,可以不添加该注解。
@Inherited
@Inherited
用于指定被它注解的 Annotation
是否可以被子类继承。它可以被用于定义其他注解。
自定义注解
@interface
是 Java 中用于创建自定义注解的关键字。使用 @interface
可以定义一个新的注解类型,并在注解中声明自定义的元素。
public @interface 注解名 {// 注解元素声明
}
在 @interface
定义的花括号中,可以声明注解的元素。
注解元素的定义类似于方法的定义,包括元素的类型、名称和可选的默认值。
注解的属性
属性定义方式与接口中的方法声明方式一样,在注解声明了方法,就同时声明了同名属性变量.
属性的赋值
注解的属性就是接口中定义的方法,定义了属性,就要给属性赋值.
- 如果定义属性时,使用 default关键字给属性默认初始化值,则使用注解时,可以不进行属性的赋值.
- 如果注解只声明了一个方法,方法名为 value,则在使用的时候, value 可以省略.
- 数组赋值时,值使用
{}
包裹,如果数组中只有一个值,则{}
省略
属性的返回值
- 基本数据类型
- String
- 枚举
- 注解
- 以上类型的数组
注解解析
在 Java 这个纯面向对象的语言中,一切的行为都是基于类的设计来描述,对注解的解析也离不开其所注解的类。
解析注解,主要还是要通过被注解的对象的(class、method、field)来获取。
类上注解解析
在注解的作用对象为@Target(ElementType.TYPE)的基础上
。
解析步骤为:
-
获取类字节码信息
-
获取该
class
上的注解信息 -
解析得到注解信息
解析属性上面的字节
在注解的作用对象为@Target(ElementType.FIELD)的基础上。
解析步骤为:
-
获取类字节码信息
-
获取字节码中的属性
-
获取该
属性
上的注解信息 -
解析得到注解信息
自定义案例
定义一个枚举
package com.jj.anno;public enum Enumeration {//星期MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY;
}
作用在字段上
package com.jj.anno;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;/*** 地址注解*/
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface AddressAnnotation {String value() default "";
}
作用在字段上
package com.jj.anno;import java.lang.annotation.Retention;
import java.lang.annotation.Target;/*** 名字注解*/
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(java.lang.annotation.ElementType.FIELD)
public @interface NameAnnotation {String name() default "";
}
作用在类型上
package com.jj.anno;import java.lang.annotation.Retention;
import java.lang.annotation.Target;/*** 拥有宠物的信息* 宠物的名字* 宠物的年龄*/
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(java.lang.annotation.ElementType.TYPE)
public @interface Pet {String name();int age();
}
作用在方法上
package com.jj.anno;import java.lang.annotation.Retention;
import java.lang.annotation.Target;@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(java.lang.annotation.ElementType.METHOD)
public @interface MethodAnnotation {String value() default "";
}
作用在类型上
package com.jj.anno;import java.lang.annotation.Retention;
import java.lang.annotation.Target;@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(java.lang.annotation.ElementType.TYPE)
public @interface Staff {//薪水int value();//岗位名字String title() default "程序员";//工龄int workAge() default 1;Enumeration weekday() default Enumeration.MONDAY;String[] hobbies();AddressAnnotation address();NameAnnotation nickname();Pet pet();
}
使用注解
package com.jj.bean;import com.jj.anno.*;@Pet(name = "jj", age = 18)
@Staff(value = 180000,title = "IT 工程师",workAge = 10,weekday = Enumeration.MONDAY,address = @AddressAnnotation("上海市"),nickname = @NameAnnotation(name = "云溪"),pet = @Pet(name = "jj", age = 18),hobbies = {"打游戏", "看电影"})
public class MyPerson {@NameAnnotation(name = "石昊")private String name;@AddressAnnotation(value = "上海市@宝山区")private String address;public String getName() {return name;}
}
注解解析
package com.jj.bean;import com.jj.anno.*;import java.lang.reflect.Field;public class Main {public static MyPerson initObj() {MyPerson myPerson = new MyPerson();Class<MyPerson> clazz = MyPerson.class;if (clazz.isAnnotationPresent(Pet.class)) {Pet pet = clazz.getAnnotation(Pet.class);System.out.println("宠物的名字:" + pet.name());System.out.println("宠物的年龄:" + pet.age());}if (clazz.isAnnotationPresent(Staff.class)) {Staff staff = clazz.getAnnotation(Staff.class);System.out.println("岗位名字:" + staff.title());System.out.println("工龄:" + staff.workAge());String[] hobbies = staff.hobbies();for (String hobby : hobbies) {System.out.println("兴趣爱好:" + hobby);}Pet pet = staff.pet();System.out.println("宠物的名字:" + pet.name());System.out.println("宠物的年龄:" + pet.age());Enumeration weekday = staff.weekday();System.out.println("工作日:" + weekday);NameAnnotation name = staff.nickname();System.out.println("staff 的 名字:" + name.name());AddressAnnotation address = staff.address();System.out.println("staff 的 地址:" + address.value());}Field[] fields = clazz.getDeclaredFields();for (Field field : fields) {field.setAccessible(true); // 确保可以访问私有字段if (field.isAnnotationPresent(NameAnnotation.class)) {NameAnnotation annotation = field.getAnnotation(NameAnnotation.class);System.out.println("名字: " + annotation.name());try {field.set(myPerson, annotation.name());} catch (IllegalAccessException e) {throw new RuntimeException(e);}}if (field.isAnnotationPresent(AddressAnnotation.class)) {AddressAnnotation annotation = field.getAnnotation(AddressAnnotation.class);System.out.println("地址: " + annotation.value());}}return myPerson;}public static void main(String[] args) {MyPerson myPerson = initObj();System.out.println("通过注解和反射创建的对象:" + myPerson.getName());}
}
相关文章:
java 知识点:注解及使用
注解 大多数时候,我们会使用注解,而不是自定义注解。注解给谁用?编译器 、给解析程序用注解不是程序的一部分,可以理解为注解就是一个标签 主要的作用有以下四方面: 生成文档,通过代码里标识的元数据生成…...
AI预测体彩排3采取888=3策略+和值012路+胆码+通杀1码测试12月13日升级新模型预测第156弹
经过100多期的测试,当然有很多彩友也一直在观察我每天发的预测结果,得到了一个非常有价值的信息,那就是9码定位的命中率非常高,已到达90%的命中率,这给喜欢打私菜的朋友提供了极高价值的预测结果~当然了,大…...
faiss数据库检索不稳定
faiss数据检索不稳定 def build_faiss_index(embeddings_vector):dim np.shape(embeddings_vector)[-1]index faiss.index_factory(dim, HNSW64, faiss.METRIC_INNER_PRODUCT)index.add(embeddings_vector)return index这个代码不稳定,构建的索引召回结果可能会不…...
Vue技术中参数传递:Props与事件的实践指南
在Vue.js中,组件间的参数传递是构建动态和交互式应用的核心。本文将深入探讨如何通过Props和事件($emit)在Vue组件间进行参数传递,并提供代码示例。 Props传递数据 Props是Vue中组件间传递数据的一种方式,它允许父组…...
C++【基础】 ---- 快速入门 C++
文章目录 前言一、有关 const 区分二、有关命名空间三、有关输入和输出四、有关缺省参数四、函数重载总结 前言 本篇文章笔者将会对 C 这么语言中必须的基础部分进行简单讲解 , 同时也作为笔者自我复习使用, 这部分是初学C 的学者不可绕过的部分 , 希望学者认真理解 ,认真领会…...
Neo4j+Neovis+Vue3:前端连接数据库渲染
Neovis(github):https://github.com/neo4j-contrib/neovis.js Neovis配置文档:neovis.js (neo4j-contrib.github.io) 一、安装Neo4j 参考文章:neo4j下载安装配置步骤-CSDN博客 二、Neovis使用 1.npm引入 ?npm ins…...
React 18
文章目录 React 18自动批处理并发特性Suspense 组件增强新 HookscreateRoot API 替代 ReactDOM.renderStrict Mode严格模式服务器端渲染改进性能优化 React 18 React 18 引入了一系列新特性和改进,旨在提升性能、改善用户体验,并简化开发流程。以下是 R…...
Java:集合(List、Map、Set)
文章目录 1. Collection集合1-1. 迭代器遍历方式1-2. 通过for循环进行遍历1-3. forEach遍历 2. List集合2-1. ArrayList底层实现原理2-2. LinkedList底层实现原理 3. Set集合3-1. HashSet 底层实现3-2. LinkedHashSet 底层实现3-3. TreeSet 4. Collection集合->总结5. Map集…...
使用秘钥登录服务器
在我们测试或生产环境中,为了服务器安全性,有时可能需要以 SSH 密钥的方式登录服务器,接下来,将演示如何通过 SSH 私钥的方式来远程服务器。 一、远程服务器生成密钥对 1、首先在目标远程服务器下生成 SSH 密钥对 ssh-keygen然…...
BFS算法题
目录 1.BFS 2.树里的宽搜 题目一——429. N 叉树的层序遍历 - 力扣(LeetCode) 题目二——103. 二叉树的锯齿形层序遍历 - 力扣(LeetCode) 题目三——662. 二叉树最大宽度 - 力扣(LeetCode) 题目四——…...
网络应用技术 实验八:防火墙实现访问控制(华为ensp)
目录 一、实验简介 二、实验目的 三、实验需求 四、实验拓扑 五、实验步骤 1、设计全网 IP 地址 2、设计防火墙安全策略 3、在 eNSP 中部署园区网 4、配置用户主机地址 5、配置网络设备 配置交换机SW-1~SW-5 配置路由交换机RS-1~RS-5 配置路由器R-1~R-3 6、配置仿…...
嵌入式现状、机遇、挑战与展望
在当今数字化浪潮中,嵌入式系统宛如一颗璀璨的明珠,熠熠生辉,深刻地渗透到了我们生活的方方面面,成为推动现代科技进步不可或缺的关键力量。从智能家居的便捷控制,到工业生产的精准运作,再到汽车的智能驾驶…...
天通卫星卡通知短信模板
文章目录 引言I 阿里云新增短信模板短信模板通知短信变量规范计费规则: 短信长度不超过70个字,按照1条短信计费;II 表设计III 实现方案引言 背景:天通卡适用于应急救灾、登山探险、海上通信、野外作业等需要稳定可靠通信的场景。 需求:天通卡充值成功通知 平台基于阿里云给…...
Unity WebGL 编译和打包说明(官方文档翻译校正)
目录 Unity WebGL 编译和打包说明WebGL 简介WebGL 浏览器兼容性 (WebGL Browser Compatibility)平台支持 (Platform Support)多线程支持限制多线程支持的因素安装 Unity Hub 并添加所需模块WebGL 开发WebGL Player 设置Resolution and PresentationResolutionWebGL TemplateSpl…...
题解 - 取数排列
题目描述 取1到N共N个连续的数字(1≤N≤9),组成每位数不重复的所有可能的N位数,按从小到大的顺序进行编号。当输入一个编号M时,就能打印出与该编号对应的那个N位数。例如,当N=3时,可…...
JAVA实战:借助阿里云实现短信发送功能
亲爱的小伙伴们😘,在求知的漫漫旅途中,若你对深度学习的奥秘、JAVA 、PYTHON与SAP 的奇妙世界,亦或是读研论文的撰写攻略有所探寻🧐,那不妨给我一个小小的关注吧🥰。我会精心筹备,在…...
高阶函数:JavaScript 编程中的魔法棒
在JavaScript的世界里,高阶函数是一种强大的工具,它允许我们将函数作为参数传递或将函数作为返回值。这种特性使得JavaScript代码更加灵活和强大。本文将深入探讨高阶函数的定义、用法以及在实际项目中的最佳实践,帮助大家更好地理解和应用这…...
Android 12.0 Launcher3从首页开始安装app功能实现
1.前言 在12.0的系统rom定制化开发中,在进行Launcher3的某些功能开发实现过程中,在某些项目中,安装的app比较多,要求在launcher3的首页开始安装 app应用,接下来就需要分析下app安装图标排序的流程,然后在实现相关的功能 2. Launcher3从首页开始安装app功能实现的核心…...
软考高级架构 - 10.5 软件架构演化评估方法
10.4 软件架构演化原则总结 本节提出了18条架构演化的核心原则,并为每条原则设计了简单而有效的度量方法,用于从系统整体层面提供实用信息,帮助评估和指导架构演化。 演化成本控制:成本小于重新开发成本,经济高效。进…...
半导体制造全流程
半导体制造是一个极其复杂且精密的过程,主要涉及将硅片加工成功能强大的芯片。以下是半导体制造的全流程概述: 1. 硅材料制备 硅提纯: 使用冶金级硅,进一步提纯为高纯度硅(电子级硅),纯度可达 …...
国科大网络协议安全期末
完整资料仓库地址:https://gitee.com/etsuyou/UCAS-Network-Protocol-Security 部分题目: 六 论述题10*220 试讨论IPv6解决了IPv4的哪些“痛点”,以及IPv6存在的安全问题试比较IPsec与SSL的安全性 五 简答题5*315 简述MAC欺骗和ARP欺骗的…...
ES动态索引——日志es索引数据按月份存储
一、定义ES索引 NoArgsConstructor AllArgsConstructor Data Accessors(chain true) Document(indexName "charge_pile_log" Constants.ES_TIME_DYNAMIC_INDEX) //(索引名称动态,前面固定,后面月份) public class ChargePileLogESDomain {…...
NLP论文速读(ICML 2024)|面相对齐大语言模型的迁移和合并奖励模型方法
论文速读|Transforming and Combining Rewards for Aligning Large Language Models 论文信息: 简介: 本文探讨了如何使大型语言模型(LLMs)与人类偏好对齐。传统的对齐方法是先从偏好数据中学习一个奖励模型,然后使用这…...
蓝桥杯我来了
最近蓝桥杯报名快要截止了,我们学校开始收费了,我们学校没有校赛,一旦报名缴费就是省赛,虽然一早就在官网上报名了,但是一直在纠结,和家人沟通,和朋友交流,其实只是想寻求外界的支持…...
Vue3+TypeScript+AntVX6实现Web组态(从技术层面与实现层面进行分析)内含实际案例教学
摘要 用Vue3+TypeScript+AntVX6实现Web组态(从技术层面与实现层面进行分析),包含画布创建、节点设计、拖拽实现(实际案例)、节点连线、交互功能,后续文章持续更新。 注:本文章可以根据目录进行导航 文档支持 AntVX6使用文档 https://x6.antv.antgroup.com/tutorial…...
【LeetCode】每日一题 2024_12_13 K 次乘运算后的最终数组 I(暴力)
前言 每天和你一起刷 LeetCode 每日一题~ 小聊两句 1、今天是 12.13 南京大屠杀国家公祭日。铭记历史,勿忘国耻。 2、今天早上去看了 TGA 年度游戏颁奖,小机器人拿下了年度最佳游戏,所有人都震惊了,大伙纷纷问到,谁…...
Plant simulation、Flexsim、Automod、Emulate3D、VisuaComponent仿真软件对比
软件名称物流系统仿真工业布局仿真动画效果数据分析优化虚拟现实/混合现实二次开发虚拟电控和PLC调试 软件行业内特殊功能Emulate3D1.物流设备模块完备,功能灵活设置,涵盖各种设备形态和运作方式 2.唯一将摩擦力、重力、阻力等物理属性融入到物流运动中&…...
深度学习day4|用pytorch实现猴痘病识别
🍨 本文为🔗365天深度学习训练营中的学习记录博客🍖 原作者:K同学啊 🍺要求: 训练过程中保存效果最好的模型参数。 加载最佳模型参数识别本地的一张图片。 调整网络结构使测试集accuracy到达88%&#x…...
批量导出工作簿中高清图片-Excel易用宝
我同事在工作簿中做了三个图表,现在需要将工作簿中的图标导出保存为高清图片,通过右键另存为保存的图片并非高清图片,其实要把Excel工作簿中的图表或图片对象导出为高清图片也很简单。 单击Excel易用宝 Plus,导出高清图片。 在导出…...
外观模式的理解和实践
外观模式(Facade Pattern)是一种常用的软件设计模式,它提供了一个统一的接口,用来访问子系统中的一群接口。该模式定义了一个高层的接口,使得子系统更容易使用。简单来说,外观模式就是通过引入一个外观角色…...
iis如何做网站/互联网广告销售是做什么的
1、下载solr :wget http://archive.apache.org/dist/lucene/solr/7.3.0/solr-7.3.0.tgz 或者去官网自己下:http://archive.apache.org/dist/lucene/solr/ 找下面图片字样点进去选版本2、解压solr 命令:tar -zxvf solr-7.3.0.tgz -C /usr/local/soft…...
wordpress 4.9中文版/手机优化
【问题描述】 STL next_permutation() 函数,常用于输出给定序列的全排列。但由于对STL next_permutation() 函数的返回值不明晰,导致一些朋友在解读使用STL next_permutation() 函数的代码时,存在困惑。例如,下方代码一、代码二的…...
沭阳做网站的公司/网站维护费用
原文:https://www.apmreports.org/episode/2019/08/22/whats-wrong-how-schools-teach-reading 传统的阅读三部曲:记住单词、通过上下文猜单词、猜不出来跳过去 memorizing wordsusing context to guess wordsskipping words they dont know Ken Goodman…...
青岛大学春季高考有网站建设吗/2345导航网址
支持正则表达式...
金融直播室网站建设/网络推广产品公司
前言 腾讯手机游戏在登录时会使用QQ或微信授权登录,此时可配置权限,包含游戏账号信息、游戏好友关系等。那么如何对腾讯游戏进行权限管理呢,有如下2种方法,分别为登录授权时配置和进入设置配置。 登录授权时配置 QQ 在QQ授权登…...
58网站 做现浇混凝土/免费关键词挖掘工具
vue.js ui组件库敏锐的用户界面 (Keen UI) A lightweight but complete collection of essential UI components written with Vue and inspired by Material Design 轻量但完整的基本UI组件集合,这些组件由Vue编写并受Material Design启发 Keen UI is designed to…...