由浅入深 带你了解 JAVA 注解

在学习注解之前,我首先来讲一讲学习注解的好处,不管下面看不看,先打个鸡血先。不过确定的是,在正常 JAVA 开发中,自己写注解是比较少的,更多的情况就是使用第三方库的注解,正因为如此,大多数开发者对于注解仅仅停留在会用的地步。试想一下,当大多数人都不会的时候你会,那么你是不是就超越的大部分人。除此之外,我还总结学习注解的3大好处:

  1. 能够读懂别人写的代码,特别是框架相关的代码;
  2. 让编程更加简洁,代码更加清晰;
  3. 让别人高看一眼,装逼利器;

说完了这些,下面就开始真干货了。

注解概念

JDK 5中引入了源代码中的注解(annotation)这一机制。 注解使得Java源代码中不但可以包含功能性的实现代码,还可以添加元数据。 注解的功能类似于代码中的注释,所不同的是注解不是提供代码功能的说明,而是实现程序功能的重要组成部分。

Java中常见注解

对于常见注解,单单看在开发 JAVA 时 IDE 弹出的那些注解就能了解到,对于JDK 的注解,我们是要非常熟悉才行。

JDK自带注解

  • @Override
  • @Deprecated
  • @Suppvisewarnings

常见第三方注解

  • Spring

    • @Autowired
    • @Service
    • @Repository
  • Mybatis

    • @InsertProvider
    • @UpdateProvider
    • @Options

注解的分类

按照运行机制分类

  • 源码注解 (注解只在源码中存在,编译成.class文件就不存在了)
  • 编译时注解 (注解在源码和.class文件中都存在)
  • 运行时注解 (在运行阶段还起作用,甚至会影响运行逻辑的注解,例如 @Autowired

按照来源分类

  • 来自JDK的注解
  • 来自第三方的注解
  • 自定义注解

元注解

给注解使用的注解

自定义注解

下面是一个典型的注解声明,大家先看个大概,下面会对定义注解的注意事项做个说明:

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Description {
    String desc();
    String author();
    int age() default 18;
}

  1. 使用@interface关键字定义注解
  2. 成员(变量)以无参数无异常方式声明
  3. 可以用default为成员指定一个默认值
  4. 注解中成员的类型是受限制的,合法的类型包括基本数据类型以及 StringClassAnnotationEnumeration
  5. 如果注解只有一个成员,则成员名应该为value(),在使用时可以忽略成员名和赋值号(=
  6. 注解类可以没有成员,没有成员的注解称为标识注解
  7. 元注解,用于对注解进行的注解,例如Description注解上面的4个注解

常用元注解

Target

标识注解的作用域,作用域有以下几种,几乎包含了 JAVA 所有的类型:

  1. ElementType.CONSTRUCTOR:构造方法
  2. ElementType.FIELD:字段
  3. ElementType.LOCAL_VARIABLE:局部变量
  4. ElementType.METHOD:方法
  5. ElementType.PACKAGE:包
  6. ElementType.PARAMETER: 参数
  7. ElementType.TYPE:类、接口

例如上面例子中标识了Description可以用于对方法和类或接口进行注解:

@Target({ElementType.PARAMETER, ElementType.TYPE})

Retention

标识注解的生命周期,有以下3种值:

  1. RetentionPolicy.SOURCE:只在源码显示,编译时会丢弃
  2. RetentionPolicy.CLASS:编译时会记录到class文件中,运行时会忽略
  3. RetentionPolicy.RUNTIME:运行时存在,可以通过反射读取

例如上面的例子中标识了Description可以记录到class文件中,但在运行时会忽略:

@Retention(RetentionPolicy.CLASS)

Inherited

标识性元注解,标识该注解允许子类进行继承。注意这里可不是注解的继承,注解之间也没有继承什么的。这里指的是如果一个类(不是接口)声明上使用了这个注解,那么当它的一个子类继承该类时,也会拥有与父类一样拥有该注解。

Documented

标识在生成 JAVA DOC 时会包含该注解

使用自定义注解

使用注解的语法:

@<注解名>(<成员名1>=<成员值1>, <成员名1>=<成员值1>, <成员名1>=<成员值1>, ...)

其中的成员名则对应了注解里的成员,例子:

@Description(desc="description", author="swifter", age=18)
public String getColor() {
    return "red";
}

上面只是个简单的例子,下面给出一个注解的详细使用方式,这也是在正常开发中会使用到的方式。对于注解,上面的代码中已经给出了一个声明,下面就写两个类来使用该注解。通过这两个例子,怎么使用注解就显而易见了。

第一个类 Person

@Description(desc="person interface", author="swifter")
public abstract class Person {
    @Description(desc="method getName", author="swifter")
    abstract String getName();
    abstract void doSomething();
}

第二个类 Child 继承自 Person

public class Child extends Person {

    @Override
    @Description(desc="child method getName", author="swifter")
    public String getName() {
        return "get child";
    }

    @Override
    public void doSomething() {
        System.out.println("do something in child class");
    }
}

解析注解

既然已经定义了注解,那么就应该考虑如何在代码中解析这个注解了。解析注解就是通过反射获取类、函数或成员上的运行时注解信息,从而实现动态控制程序运行的逻辑。
解析主要方式就是通过反射的途径,通过Class对象来获取注解,并拿到注解的字段再进行操作。例如下面的例子:

Class<Child> clazz = Child.class;

if(clazz.isAnnotationPresent(Description.class)) {
    Description description = clazz.getAnnotation(Description.class);
    System.out.println(description.desc()+" : "+description.author());
}

Method[] methods = clazz.getMethods();
for(Method method : methods) {
    if(method.isAnnotationPresent(Description.class)) {
        Description description = method.getAnnotation(Description.class);
        System.out.println(description.desc()+" : "+description.author());
    }
}

运行之后客户端会给出如下结果:

person interface : swifter
child method getName : swifter

其中第一行显示的是类上面的注解信息,第二行则显示的是方法上的注解信息。对于这两个注解的对象,在代码中可以看到,都是通过反射的形式拿到的。虽然说反射的方式在效率上比较慢,但是想想注解带来的优点,掌握注解还是有很大必要的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 什么是注解(Annotation):Annotation(注解)就是Java提供了一种元程序中的元素关联任何信息和...
    九尾喵的薛定谔阅读 3,149评论 0 2
  • 本文章涉及代码已放到github上annotation-study 1.Annotation为何而来 What:A...
    zlcook阅读 29,131评论 15 116
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,637评论 18 139
  • 日记的最近更新日期停在6月25号,已经一个多月没写日记了。于是,长时间的积累,给了自己一个下笔的借口。 6月25,...
    遥途阅读 406评论 1 3
  • 多样化的现代生活带给我们很多新鲜感的同时,也使很多人得了效应并发症----孤独症。 丰富光鲜的朋友圈背后更体现了一...
    阿菜Choi阅读 470评论 1 1