Java源码赏析(四)Java常见注解
元注解
@Target :标识注解的目标,默认为所有
ElementType.TYPE(用于类)
ElementType.FIELD(用于域,包括enum)
ElementType.METHOD(用于方法)
ElementType.PARAMETER(用于正式参数)
ElementType.CONSTRUCTOR(类型构造方法)
ElementType.LOCAL_VARIABLE(用于本地变量)
ElementType.ANNOTATION_TYPE(用于注解)
ElementType.PACKAGE(用于包)
ElementType.TYPE_PARAMETER(java8,用于输入参数)
ElementType.TYPE_USE (java8,用于类型)
@Retention:注解保留到哪
- RetentionPolicy.SOURCE (只保留到源码,编译为.class文件时忽略)
- RetentionPolicy.CLASS (只保留到.class文件,运行时忽略,default默认)
- RetentionPolicy.RUNTIME (运行时保留,可以通过反射机制读取注解的信息。)
@Documented:将此注解保存在 Javadoc 中
@Inherited:允许子类继承父类的注解
@Repeatable:用于重复使用注解(Java8新增)
常用注解
一、@Override
常用于重写父类的方法,或实现接口的方法,若不满足则报错。
package java.lang; import java.lang.annotation.*; /**
* 使用在方法上,只存在在源码上
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}
二、@Deprecated
常用于标注一个方法被弃用。
package java.lang; import java.lang.annotation.*;
import static java.lang.annotation.ElementType.*; /**
* 记录JavaDoc,保留至运行时,可使用在构造方法,属性(包括枚举),本地变量,方法,包,公共参数和类
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, PARAMETER, TYPE})
public @interface Deprecated {
}
三、@SuppressWarnings
常用于忽略特定警告,@SuppressWarnings("unchecked", "deprecation")可以同时忽略多个警告。
package java.lang; import java.lang.annotation.*;
import static java.lang.annotation.ElementType.*; /**
* 在类型、域、方法、正是参数、构造方法、本地变量上使用,只存在于源码
*/
@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE})
@Retention(RetentionPolicy.SOURCE)
public @interface SuppressWarnings {
/**
* 一般取 unchecked/deprecation,表示忽略 unchecked/deprecation 警告信息
*/
String[] value();
}
四、@SafeVarargs (Java7)
常用于抑制varargs相关的未检查警告,一般只能用于static、final方法
package java.lang; import java.lang.annotation.*; /**
* 能记录JavaDoc,保留至运行时,使用在构造方法和方法上
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD})
public @interface SafeVarargs {}
五、@FunctionalInterface (Java8)
编译器检测接口是否符合函数式接口定义
package java.lang; import java.lang.annotation.*; /**
* 能记录JavaDoc,存在于运行时,只能使用在类上
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface FunctionalInterface {}
自定义注解
首先定义一个注解
package com.example.demo.util.annotation; import java.lang.annotation.*; @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
@Documented
@Inherited
public @interface MyAnnotation {
String value() default "001";
}
可以看到,在Student的sNo属性上使用了该注解
package com.example.demo.util.annotation; import lombok.Data;
import lombok.AllArgsConstructor;
import lombok.NoArgsConstructor; @Data
@AllArgsConstructor
@NoArgsConstructor
public class Student {
@MyAnnotation
private String sNo; private String sName;
}
处理注解
package com.example.demo.util.annotation; import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Objects; public class AnnotationTest { public static void main(String[] args) {
//创建一个学号为 002 的学生 kw
Student student = new Student("002", "kw"); //获取学生对象对应的类
Class<? extends Student> clazz = student.getClass();
//获取学生类的属性
Field[] declaredFields = clazz.getDeclaredFields();
/** 循环判断是否存在注解@MyAnnotation
* 若存在则获取该属性的set方法
* 并使用该方法将值改为@MyAnnotation的value值
*/
@MyAnnotation的value
for (Field field: declaredFields) {
if (field.isAnnotationPresent(MyAnnotation.class)) {
MyAnnotation annotation = field.getAnnotation(MyAnnotation.class);
if (Objects.nonNull(annotation)) {
String name = field.getName();
try {
Method setMethod = clazz.getDeclaredMethod(
"set" + name.substring(0, 1).toUpperCase() + name.substring(1), String.class);
String annotationValue = annotation.value();
setMethod.invoke(student, annotationValue);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
} }
}
//打印
System.out.println(student);
}
} /** 输出
Student(sNo=001, sName=kw)
*/
Java源码赏析(四)Java常见注解的更多相关文章
- Java源码赏析(三)初识 String 类
由于String类比较复杂,现在采用多篇幅来讲述 这一期主要从String使用的关键字,实现的接口,属性以及覆盖的方法入手.省略了大部分的字符串操作,比如split().trim().replace( ...
- Java源码赏析(六)Class<T> 类
目的 Class 类是每一个程序员都必须了解的,也是使用反射机制的基础. 这篇文章将Class 类的公共方法大致介绍了一遍(省略了安全.枚举.断言.注解相关代码). 代码 package java.l ...
- Java源码赏析(五)再识 String 类
在 Java源码赏析(三)初识 String 类 中,我们已经大概理解了String的接口,接下来我们描述一下String的常用工具方法. /** * 为了精简的String结构,之前提到的方法省 ...
- 【java集合框架源码剖析系列】java源码剖析之java集合中的折半插入排序算法
注:关于排序算法,博主写过[数据结构排序算法系列]数据结构八大排序算法,基本上把所有的排序算法都详细的讲解过,而之所以单独将java集合中的排序算法拿出来讲解,是因为在阿里巴巴内推面试的时候面试官问过 ...
- Java源码赏析(二)Java常见接口
一.Comparable接口 package java.lang; import java.util.*; public interface Comparable<T> { /** * i ...
- Java 源码赏析 - java.lang - Void
被人鄙视了,于是也来读读源码... package java.lang; /** * The Void class is an uninstantiable placeholder class to ...
- Java源码赏析(一)Object 类
写这个系列的原因,其实网上已经有无数源码分析的文章了,多一篇不多,少一篇不少,但为什么还要写这部分文章呢?于私,其一,上班族已经很久没有打过完整的一整段有意义的话,算是锻炼个人的书写.总结能力,其二, ...
- Java源码赏析(六)Java String 三顾
在大致了解了String之后,可能有的读者发现了,我们并没有谈到CharSequence接口. 原因是在这一节,CharSequence要和StringBuilder(Java1.5).StringB ...
- 24点扑克牌游戏——(含java源码)(GUI实现)
给出四个数字,要求,在其间添加运算符和括号,使得计算结果等于24. 括号的放置即为决定哪几个数先进行计算.所以,我们先确定首先进行计算的两个相邻的数,计算完成后,就相当于剩下三个数字,仍需要在它们之间 ...
随机推荐
- Spring Security自定义认证页面(动态网页解决方案+静态网页解决方案)--练气中期圆满
写在前面 上一回我们简单分析了spring security拦截器链的加载流程,我们还有一些简单的问题没有解决.如何自定义登录页面?如何通过数据库获取用户权限信息? 今天主要解决如何配置自定义认证页面 ...
- 为什么我们需要Logstash,Fluentd等日志摄取器?
前文传送门:Logging with ElasticSearch, Kibana, ASP.NET Core and Docker 疑问:既然应用能直接向ElasticSearch写日志,为什么我们还 ...
- TX-LCN 分布式事务框架
第十章 TX-LCN 分布式事务框架 (Spring Cloud 高级) 一. 什么是分布式事务 分布式事务是指事务的参与者.支持事务的服务器.资源服务器以及事务管理器分别位 于不同的分布式系统的不同 ...
- Selenium中核心属性以及方法
一.操作定位元素 selenium提供了定位元素的API,这些方法都被定义在webDriver类中,需要以find开头, 例如:find_Element_by_id('')
- Picker 组件的设计与实现
前言 今天的主题是 Picker 组件的设计与实现,Picker 组件是 NutUI 的一个拾取器组件,它用于显示一系列的值集合,用户可以滚动选择集合中一项,也可以支持多个系列的值集合供用户分别选择. ...
- sublime Text 3安装 Sublime Package Control(这个可以用于安装各种插件)时显示默认安装的c盘内存不够的解决方案
首先先关闭st3 之后在安装的路径下创建Data文件夹,然后打开st3(sublime Text 3简写)CTRL+`打开命令行输入以下内容,直接回车,等待下载完成就可以了import urllib. ...
- php利用快递100接口获取物流信息
PHP使用CURL调用快递100接口查询运单信息 类代码如下: <?php/** * 快递100接口调用类 * @author 齐云海 * date: 2019/05/29 */ class E ...
- 《神经网络的梯度推导与代码验证》之CNN的前向传播和反向梯度推导
在FNN(DNN)的前向传播,反向梯度推导以及代码验证中,我们不仅总结了FNN(DNN)这种神经网络结构的前向传播和反向梯度求导公式,还通过tensorflow的自动求微分工具验证了其准确性.在本篇章 ...
- Codeforces1312D Count the Arrays 组合数学
题意 给你\(n\)和\(m\),问满足以下条件的数列的个数: 数列长度为\(n\) 数列值域范围为\(\left[1,m\right]\) 数列有且仅有一对相等的数 数列是单峰数列(先严格递增后严格 ...
- 小程序开发-小程序tabBar不显示的原因分析
在尝试小程序开发时,tabBar这个是个非常常见的组件,但是今天在进行开发时,设置了TabBarb并没有显示,被这个问题困扰了近半小时,现在将排查问题后所得到的经验记录下来. 首先 如果tabBar不 ...