java 注解(annotation)

java 注解(annotation)

Java 5.0 版本之后引入注解,至此就被广泛使用。

何为注解

什么是注解呢?简单说,注解就是描述数据的数据(Data that describes other data),即 元数据
就像注释一样,注解也能对源代码进行描述;除描述外注解还能实现对数据的分类;
在元数据方面,XML 被大量使用(过去和现在,估计未来也是如此)。
XML 和注解的共同点:都只是描述性内容,本身不具备任何业务处理能力
XML 和注解的不同点:XML 和代码是松耦合的;注解和代码在物理距离上更近(紧耦合)。
很多情况下,XML 配置其实就是为了分离代码和配置而使用的。但有些场景,我们更希望与代码紧密结合。

最常见的两个 java 注解

  • @Override
    @Override
    public String toString() {
        return super.toString();
    }

点进 @Override 后去掉注释是这样的:

package java.lang;

import java.lang.annotation.*;

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}

这妥妥就是一个空的注解,没有属性!没有方法!
Override 这个单词的中文翻译就是覆盖,@Override 注解标识了其所标注的方法是在其父类或者其实现的接口中声明的。被 @Override 标注的方法是覆盖或实现已经被声明过的方法。
既然,本身不具备任何业务处理能力,那 @Override 又是如何起效的呢?
对于 @Override 来说,是编译器在检查所有背其标注的方法,检查父类或者实现的接口里是否有该方法的声明。若不存在,则编译器报错,提醒错误。

  • @Deprecated
    @Deprecated 是长成这样子的:
package java.lang;

import java.lang.annotation.*;
import static java.lang.annotation.ElementType.*;

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, PARAMETER, TYPE})
public @interface Deprecated {
}

没错!这还是一个空注解。
程序员使用这个注解是提醒调用者,被其标注的方法是不安全的或者有更好替代方案的,是被废弃的。调用方若使用了被 @Deprecated 标注的方法,在编译时会收到一个警告。

如何创建一个自定义注解

还是直接端上两个自定义注解再解释吧
只有一个属性的自定义注解

package com.hxx.annotation;

import java.lang.annotation.*;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Author {
    String value();
}

多个属性的自定义注解

package com.hxx.annotation;

import com.hxx.enums.UserType;

import java.lang.annotation.*;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface AnnotationDemo {
    long time() default 0;

    int count() default 0;

    String name() default "XiaoMing";

    UserType userType() default UserType.SYSTEM_ADMIN;

}

其中,UserType 是一个枚举

package com.hxx.enums;

public enum UserType {
    SYSTEM_ADMIN("系统管理员"),
    TOURISTS("游客");
    private String description;

    UserType(String description) {
        this.description = description;
    }

    public String getDescription() {
        return description;
    }
}

元注解(专门注解其他注解的注解)

J2SE5.0版本在 java.lang.annotation 提供了四种元注解:

注解名 注解含义
@Documented 注解是否将包含在 JavaDoc 中
@Retention 注解的生命周期
@Target 注解能被用在什么地方
@Inherited 是否子类允许继承该注解

@Documented

被 @Documented 标注的注解,注解信息会被添加在 JavaDoc 中

@Retention

@Retention 定义了其标记注解的生命周期。
可选的值有:

  • RetentionPolicy.SOURCE
    在编译阶段丢弃,只作用于编译阶段,不会写入字节码。比如:@Override, @SuppressWarnings。
  • RetentionPolicy.CLASS
    在类加载的时候丢弃,字节码文件的处理中有用。注解默认使用这种方式
  • RetentionPolicy.RUNTIME
    始终不会丢弃,运行期也保留该注解,因此可以使用反射机制读取该注解的信息。自定义注解时常用

@Target

@Target 表示该注解用于什么地方。如果不明确指出,该注解可以放在任何地方。
可选的值有:

描述场景
ElementType.TYPE 类、接口或枚举声明
ElementType.FIELD 属性(变量)声明
ElementType.METHOD 方法声明
ElementType.PARAMETER 形参声明
ElementType.CONSTRUCTOR 构造函数声明
ElementType.LOCAL_VARIABLE 局部变量声明
ElementType.ANNOTATION_TYPE 用于标注注解(如元注解)
ElementType.PACKAGE 包声明

java 8.0 又新增两个可选值:ElementType.TYPE_PARAMETER、ElementType.TYPE_USE。
可以同时指定多个取值,互相不影响(因为@Target 的属性是个ElementType数组)。

@Inherited

@Inherited 定义该注释和子类的关系,仅对类起效。

注解的属性

注解(Annotations)只支持基本类型、String及枚举类型作为其属性,且所有的属性被定义成方法,并允许提供默认值(default关键字)。

注解的使用

上面定义的 @AnnotationDemo 和 @Author 的用法示例如下:

package com.hxx.model;

import com.hxx.annotation.*;
import com.hxx.enums.UserType;

@Author("houxx")
public class Person {
    @AnnotationDemo(time = 1, count = 2, name = "admin", userType = UserType.SYSTEM_ADMIN)
    public void work() {
        System.out.println("Now,to work!");
    }

注解名后跟小括号标注其属性。标注方式:属性名 = 值,多个属性以 , 分隔。有默认值的属性可不赋值。
若是注解只有一个属性,可以直接命名为value,使用时无需再标明属性名。如:@Author和元注解。

适合注解的场景

上文明确说明了,注解不具备任何业务处理能力,这样的元数据似乎并不非常重要。
但事实并非如此,各种开发框架里都大量的使用了注解(如 Spring、Hibernate)。
注解的声明和注解的处理逻辑是分不开的。后者常常以 AOP(面向切面编程) 的形式实现。
AOP 中 annotation 的用法在下篇文章里。

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

推荐阅读更多精彩内容