高级工程师-Java注解

前言

代码,就是我们身为程序员的名片。

简洁,优雅,统一,是我们的追求。

优秀的代码,会给浏览者一种艺术的美感。如DL大神的JUC包,感兴趣的小伙伴,可以研究一下。

那么日常中,各位看到的优秀代码,有着哪些特点呢?充分利用的工具类(lang3,lombok,Validation等等),完善的注解,统一的代码规范等等。还有的,就是Java语言的诸多高级特性(lambda,stream,io等)。

Java语言中,有三个特性,是高级工程师不可或缺的:

  • 注解
  • 反射
  • 泛型

如果代码中,存在这些东西,那么即使应用得还不够合理,也能够从侧面证明这位程序员的技术追求。

这三点是初级工程师很难掌握的,因为缺乏了解与需求(或者说想不到对应的需求)。而高级工程师为了给出更加具有通用性,业务无侵入的代码,就常常需要与这些特性打交道。

在不断积累后的今天,我觉得我可以尝试写一写自己对这些特性的认识了。

今天就从注解开始,阐述我对高级工程师的一些编码认识。

简介

我发现很多小伙伴总是在喜欢记忆一些注解的功能,比如表示非空的@NotNull等。

这里,我要从功能与原理角度说明两点:

  • 功能:注解是一种“增强型”的注释。只不过相对于只能给人看的注释,注解可以给电脑(JVM,程序等)看。
  • 原理:注解的底层是Annotation接口的继承者。只不过相对于日常使用的接口,注解需要使用@interface,但是编译的结果依旧是接口继承(如TestAnnotation extend Annotation)。

请大家牢记上面两点,这是有关注解认识的绝对核心

只要大家抓住这两个角度去认识注解,那么很快就可以成为注解达人。后续很多阐述都会从这两个角度,去为大家解释。如为什么人们常说注解是无法继承的,为什么需要元注解等等。

注解的目录结构

其实可以看到,JDK中有关注解的内容很少,非常适合作为三大特性的入门啊。因为注解的实现基础是存在于JVM中的,JDK只是提供了对应的工具。

Annotation接口

上面提到注解的底层是接口,这里以图为证。

注意,仔细看这个接口的注释。注释中明确提出,虽然注解的本质是接口。但是直接引用Annotation接口,是无法实现注解功能的。

元注解

简介

通俗来说,元注解就是注解的注解。

首先元注解,是Java自带的预置注解。从这个角度,需要与@XXX修饰的自定义注解进行区分。

站在功能上来说,元注解就是专门修饰注解的“注释”,用来告诉编译器,虚拟机,相关的信息(如运行时间,目标对象)。

站在原理上来说,元注解也是注解,也是使用了@interface(底层依旧是继承Annotation接口)。

不过注解,在底层实现已经继承Annotation接口,那么就无法通过继承接口的方式(Java不支持多重继承),来保存元注解的信息(尤其这个信息往往不止一类)。那么注解的元注解信息是如何保存,并交给计算机的呢?答案就是通过RuntimeVisibleAnnotations进行相关信息的保存的。以下就是对DynamicPropertyVerification注解反编译的结果,重点在于反编译结果的最后一段。


Classfile /D:/IDEA_Project/IdeaProjects/learning/demo/target/classes/tech/jarry/learning/demo/common/anno/DynamicPropertyVerification.class
Last modified Apr 12, 2020; size 899 bytes
MD5 checksum 72657e8b89f0de070bf7085b0dd975da
Compiled from "DynamicPropertyVerification.java"
public interface tech.jarry.learning.demo.common.anno.DynamicPropertyVerification extends java.lang.annotation.Annotation
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_INTERFACE, ACC_ABSTRACT, ACC_ANNOTATION
Constant pool:
#1 = Class #28 // tech/jarry/learning/demo/common/anno/DynamicPropertyVerification
#2 = Class #29 // java/lang/Object
#3 = Class #30 // java/lang/annotation/Annotation
#4 = Utf8 message
#5 = Utf8 ()Ljava/lang/String;
#6 = Utf8 AnnotationDefault
#7 = Utf8 property verification fail
#8 = Utf8 groups
#9 = Utf8 ()[Ljava/lang/Class;
#10 = Utf8 Signature
#11 = Utf8 ()[Ljava/lang/Class<*>;
#12 = Utf8 payload
#13 = Utf8 ()[Ljava/lang/Class<+Ljavax/validation/Payload;>;
#14 = Utf8 SourceFile
#15 = Utf8 DynamicPropertyVerification.java
#16 = Utf8 RuntimeVisibleAnnotations
#17 = Utf8 Ljava/lang/annotation/Documented;
#18 = Utf8 Ljava/lang/annotation/Target;
#19 = Utf8 value
#20 = Utf8 Ljava/lang/annotation/ElementType;
#21 = Utf8 FIELD
#22 = Utf8 Ljava/lang/annotation/Retention;
#23 = Utf8 Ljava/lang/annotation/RetentionPolicy;
#24 = Utf8 SOURCE
#25 = Utf8 Ljavax/validation/Constraint;
#26 = Utf8 validatedBy
#27 = Utf8 Ltech/jarry/learning/demo/common/anno/DynamicPropertyVerificationValidator;
#28 = Utf8 tech/jarry/learning/demo/common/anno/DynamicPropertyVerification
#29 = Utf8 java/lang/Object
#30 = Utf8 java/lang/annotation/Annotation
{
public abstract java.lang.String message();
descriptor: ()Ljava/lang/String;
flags: ACC_PUBLIC, ACC_ABSTRACT
AnnotationDefault:
default_value: s#7
public abstract java.lang.Class<?>[] groups();
descriptor: ()[Ljava/lang/Class;
flags: ACC_PUBLIC, ACC_ABSTRACT
AnnotationDefault:
default_value: []Signature: #11 // ()[Ljava/lang/Class<*>; public abstract java.lang.Class<? extends javax.validation.Payload>[] payload();
descriptor: ()[Ljava/lang/Class;
flags: ACC_PUBLIC, ACC_ABSTRACT
AnnotationDefault:
default_value: []Signature: #13 // ()[Ljava/lang/Class<+Ljavax/validation/Payload;>;
}
SourceFile: "DynamicPropertyVerification.java"
RuntimeVisibleAnnotations:
0: #17()
1: #18(#19=[e#20.#21])
2: #22(#19=e#23.#24)
3: #25(#26=[c#27])

最后一段,通过RuntimeVisibleAnnotations,保存了所需要的元注解信息。

如果对JVM底层原理有了解的小伙伴,应该对RuntimeVisibleAnnotations不陌生。不了解的小伙伴,可以查看Class RuntimeVisibleAnnotations

预置注解

元注解

元注解是Java自带的,主要分为:

  • @Rentention:表示目标注解的保持策略。其value为RetentionPolicy。如果目标注解没有使用该注解,则默认使用RetentionPolicy.CLASS
  • @Target:表示目标注解的应用目标类型。其value为ElementType。如果目标注解没有使用该注解,则目标注解可以用于除了TYPE_PARAMETER和TYPE_USE以外的任何地方(这两个类型都是Java8新添加的)。
  • @Documented:表示目标注解可以出现在JavaDoc中。
  • @Repeatable:表示目标注解可以在同一位置,重复使用。
  • @Inherited:表示目标注解可以随着所修饰的类的继承关系,被子类继承。

@Retention

源码:


@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Retention {
/**
* Returns the retention policy.
* @return the retention policy
*/
RetentionPolicy value();
}

通过RetentionPolicy枚举表示目标注解的保持策略。


public enum RetentionPolicy {
/**
* 目标注解会在编译期丢失
*/
SOURCE, /**
* 默认行为。虽然目标注解会通过编译,保存至.class文件中,但是JVM不会在运行时识别该注解。
*/
CLASS, /**
* 常用行为。目标注解会保存至.class文件中,JVM会在运行时识别,并记录该注解。所以可以通过反射获取对应的信息。
* 详见 java.lang.reflect.AnnotatedElement
*/
RUNTIME
}

为了便于大家理解,这里再举一些例子。这里挑选一些Java自带的,不用大家再去自己写demo,增加认知负荷:

  • @Retention(RetentionPolicy.SOURCE):如@Override注解,由于该注解只是用于进行代码检测,所以只要存在于源码中即可,故选择RetentionPolicy.SOURCE。类似的还有@SuppressWarnings注解等。
  • @Retention(RetentionPolicy.CLASS):涉及注解处理器,所以实例很少。可以查看自定义注解之编译时注解(RetentionPolicy.CLASS)(一)
  • @Retention(RetentionPolicy.RUNTIME):如@Deprecated,由于该注解需要在运行时提示用户注解修饰的方法,类等已经过时,所以需要JVM中有对应“注释”信息,故采用RetentionPolicy.RUNTIME。类似的还有@Repeatable等。

@Target


@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Target {
/**
* Returns an array of the kinds of elements an annotation type
* can be applied to.
* @return an array of the kinds of elements an annotation type
* can be applied to
*/
ElementType[] value();
}

通过ElementType枚举表示目标注解的应用目标类型。


public enum ElementType {
/** 类,接口(包括注解,即Annotation接口),或者枚举类型 */
TYPE, /** 属性 (包括枚举常量,枚举常量示例:Retention.SOURCE) */
FIELD, /** 方法 */
METHOD, /** 形参(形式参数) */
PARAMETER, /** 构造器 */
CONSTRUCTOR, /** 本地变量 */
LOCAL_VARIABLE, /** 注解类型 */
ANNOTATION_TYPE, /** 包 */
PACKAGE, /**
* 类型参数(针对数据类型)
* @since 1.8
*/
TYPE_PARAMETER, /**
* 类型(功能域包含PARAMETER与TYPE_PARAMETER)
* @since 1.8
*/
TYPE_USE
}

这里不会一一举例,只会点出重点:

  • TYPE_PARAMETER与TYPE_USE是Java8新增加的。所以使用Java7的小伙伴要注意。
  • ElementType.TYPE涵盖范围很广泛,在不知用哪个时,可以先用这个。

@Documented

默认情况下,注解是不出现在 javadoc 中的。通过给目标注解加上 @Documented 元注解,能使目标注解出现在 javadoc 中。

从源码可以看出,@Documented是一个没有任何成员的标记注解。

@Repeatable

@Repeatable注解的使用,引用一个不错的demo


package com.zejian.annotationdemo;
import java.lang.annotation.*;/**
* Created by zejian on 2017/5/20.
*/
@Target({ElementType.TYPE,ElementType.FIELD,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(FilterPaths.class)
public @interface FilterPath {
String value();
} @Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface FilterPaths {
FilterPath[] value();
} @FilterPath("/web/update")
@FilterPath("/web/add")
@FilterPath("/web/delete")

上述代码,其实分为两个部分:

  • 使用@Repeatable注解,使得其修饰的@FilterPath,可以在目标上重复标记(便于设置不同的成员变量)。
  • 通过@FilterPaths注解(包含成员变量-FilterPath[] value ()),将@FilterPath集中到@FilterPaths中,便于后续逻辑处理。

@Inherited

@Inherited同样是只能修饰注解的元注解,它所标注的目标注解具有继承性。

这里解释一下这个继承性,这并不是注解间的继承。而是指目标注解可以随着类的继承,而被子类继承。简单说,就是目标注解修饰的类,其后代类也会被该注解标注(可以通过getAnnotation方法获取)。

这里不再赘述,感兴趣的小伙伴,可以查看Java 注解(Annotation)中的相关示例。

功能注解

Java预置的功能注解,主要分为:

  • @Override:该注解修饰的目标方法,必须是重写基类方法,或实现对应接口方法,否则编译器会报错。
  • @Deprecated:该注解修饰的目标,表示已经过时,不推荐使用。编码时,使用该注解的目标,会有划线提示。
  • @SuppressWarnings:该注解修饰的目标,将会忽略某些异常(由注解的value指定),从而通过编译器编译。
  • @SafeVarargs:该注解修饰的构造函数(只能修饰构造函数),将会忽略可变参数带来的警告。该注解于Java7引入。
  • @FunctionalInterface:该注解修饰的接口,为函数式接口。如java.util.function下的Consumer接口,作为一个函数式接口,被该注解修饰(函数式接口不一定有该注解修饰,但被该注解修饰的接口,一定是函数式接口)。

自定义注解

到了这里,大家应该对注解不再陌生了。

而在日常开发中,我们常常需要自定义开发一些注解。

自定义注解分为以下步骤:

  1. [必选] 使用@interface来构建自定义注解。一般在创建自定义注解的同时,就达成了该要求。
  2. [可选] 使用@Target元注解。通过该注解,确定自定义注解的作用目标类型。注意:如果目标注解没有使用该注解,则目标注解可以用于除了TYPE_PARAMETER和TYPE_USE以外的任何地方(这两个类型都是Java8新添加的)。
  3. [可选] 使用@Retention元注解。通过该注解,明确自定义注解的生命周期,或者说自定义注解作用域。如果目标注解没有使用该注解,则默认使用RetentionPolicy.CLASS
  4. [可选] 添加成员变量。格式为“long value() default 1000L;”,与Java8的接口成员变量非常类似。注意:注解的成员变量只能采用无参方法表示。并且注解的成员变量,只能采用基本数据类型(char,boolean,byte、short、int、long、float、double)和String、Enum、Class、annotations数据类型,以及这一些类型的数组。
  5. [可选] 使用自定义注解。自定义注解的使用领域很多,主要分为两个方向:
    • 利用已有框架,不需要自己实现相关逻辑,自定义注解多作为标记注解。如配合SpringBoot的注解,形成自己的注解(相关的逻辑由SpringBoot自己处理)
    • 利用已有框架,需要自己实现部分逻辑(不涉及反射),但需要关联已有框架,并实现对应接口。如Validation框架的自定义校验注解,感兴趣的小伙伴,可以查看我之前写的Validation框架的应用
    • 可选择已有框架,需要自己实现诸多逻辑。如在AOP中,我们常常需要通过反射,获取自定义注解的信息(如参数等),或者自定义注解修饰的目标的信息(如参数,方法名等)。这部分,我会在后续的反射部分详细说明。

总结

简单总结一下,本文主要描述了:

  1. 注解是什么:增强型的注释,本质是接口
  2. 元注解是什么:注解的注解,作用是为了标识目标注解。包括@Target,@Retention,@Documented,@Repeatable,@Inherited.
  3. 预置注解是什么:JDK自带的经典功能注解,如@Override,@Deprecated,@SuppressWarnings,@SafeVarargs,@FunctionalInterface。
  4. 自定义注解如何实现:主要分为五步,但是其中必要的步骤,就一步:使用@interface来构建自定义注解。

至此,Java注解的内容就基本展现了。

最后,还是强调两个方面:

  1. 注解就是增强型的注释(可被计算机识别的注释),本质是接口。把握住这两点,就非常好理解注解与它的各种规则,行为。
  2. 注解本身并没有任何功能(因为它只是注释,本质也只是接口),需要其他代码支撑,它才能体现价值。

希望对大家有所帮助,还有不清楚的地方,可以查看下列参考目录。

愿与诸君共进步。

附录

参考

高级工程师-Java注解的更多相关文章

  1. Java注解

    Java注解其实是代码里的特殊标记,使用其他工具可以对其进行处理.注解是一种元数据,起到了描述.配置的作用,生成文档,所有的注解都隐式地扩展自java.lang.annotation.Annotati ...

  2. 19.Java 注解

    19.Java注解 1.Java内置注解----注解代码 @Deprecated                                    //不推荐使用的过时方法 @Deprecated ...

  3. Java注解入门

    注解的分类   按运行机制分:   源码注解:只在源码中存在,编译后不存在 编译时注解:源码和编译后的class文件都存在(如@Override,@Deprecated,@SuppressWarnin ...

  4. java注解(Annotation)解析

    注解(Annotation)在java中应用非常广泛.它既能帮助我们在编码中减少错误,(比如最常见的Override注解),还可以帮助我们减少各种xml文件的配置,比如定义AOP切面用@AspectJ ...

  5. JAVA 注解的几大作用及使用方法详解

    JAVA 注解的几大作用及使用方法详解 (2013-01-22 15:13:04) 转载▼ 标签: java 注解 杂谈 分类: Java java 注解,从名字上看是注释,解释.但功能却不仅仅是注释 ...

  6. attilax.java 注解的本质and 使用最佳实践(3)O7

    attilax.java 注解的本质and 使用最佳实践(3)O7 1. 定义pojo 1 2. 建立注解By eclipse tps 1 3. 注解参数的可支持数据类型: 2 4. 注解处理器 2 ...

  7. paip.java 注解的详细使用代码

    paip.java 注解的详细使用代码 作者Attilax 艾龙,  EMAIL:1466519819@qq.com 来源:attilax的专栏 地址:http://blog.csdn.net/att ...

  8. JAVA 注解的几大作用及使用方法详解【转】

    java 注解,从名字上看是注释,解释.但功能却不仅仅是注释那么简单.注解(Annotation) 为我们在代码中添加信息提供了一种形式化的方法,是我们可以在稍后 某个时刻方便地使用这些数据(通过 解 ...

  9. 框架基础——全面解析Java注解

    为什么学习注解? 学习注解有什么好处? 学完能做什么? 答:1. 能够读懂别人写的代码,特别是框架相关的代码: 2. 让编程更加简洁,代码更加清晰: 3. 让别人高看一眼. spring.mybati ...

随机推荐

  1. F版本SpringCloud 2—什么是SpringCloud?SpringCloud版本选择

    引言:搭建微服务架构就像是买电脑,使用SpringCloud就是在买品牌机. 前言 昂,美好的天气里,不想直接说技术,给小伙伴萌看看傍晚的天空吧. -- 能找到天上的北极星吗? 上一篇文章中,通过一个 ...

  2. HTML节点操作

    HTML节点操作 HTML节点的基本操作,添加节点,替换节点,删除节点,绑定事件,访问子节点,访问父节点,访问兄弟节点. 文档对象模型Document Object Model,简称DOM,是W3C组 ...

  3. 第四周java实验

    实验四 类与对象的定义及使用 实验时间 2018-9-20 1.实验目的与要求 (1) 理解用户自定义类的定义: 类是具有相同属性和行为的一组对象的集合.java中,用构造器构造并初始化对象. 类是构 ...

  4. 【2019沈阳网络赛】G、Special necklace——自闭的物理题

    这道题让我差点怀疑自己高考没考过物理 题意中 he measures the resistance of any two endpoints of it, the resistance values ...

  5. 2,MapReduce原理及源码解读

    MapReduce原理及源码解读 目录 MapReduce原理及源码解读 一.分片 灵魂拷问:为什么要分片? 1.1 对谁分片 1.2 长度是否为0 1.3 是否可以分片 1.4 分片的大小 1.5 ...

  6. 用序列到序列和注意模型实现的:Translation with a Sequence to Sequence Network and Attention

    In this project we will be teaching a neural network to translate from French to English. 最后效果: [KEY ...

  7. 【3D】PoseCNN姿态检测网络复现过程记录

    最近在研究室内6D姿态检测相关问题,计划在PoseCNN网络基础上进行改进实现.但是在第一步的复现过程中踩了无数的坑,最终成功运行了demo,但目前数据集train还是遇到了一些问题.有问题欢迎一起交 ...

  8. Python第三方包之PrettyTable

    Python第三方包之PrettyTable 可以让我们将数据用表格的方式展示出来 安装方式 pip install PrettyTable 测试是否安装成功 使用方法与对比 增加一条数据 先简单的看 ...

  9. HDU-1421-搬寝室(01背包改编版)

    搬寝室是很累的,xhd深有体会.时间追述2006年7月9号,那天xhd迫于无奈要从27号楼搬到3号楼,因为10号要封楼了.看着寝室里的n件物品,xhd开始发呆,因为n是一个小于2000的整数,实在是太 ...

  10. 28. 实现 strStr()

    d地址:https://leetcode-cn.com/problems/implement-strstr/ <?php /** 实现 strStr() 函数. 给定一个 haystack 字符 ...