Java Annotation

Java Annotation

Java 1.5中开始引入的Annotation,类似于注释的一种技术,参考了一些网上的译法,姑且译成注解吧。

我们在开发中,用得最多的Annotation莫过于@Override了。大家天天用,可能很多同学却没有关注过其背后的细节,我们看一下它的定义:

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

在Android源码中,这个注解定义在libcore/luni/src/main/java/java/lang/Override.java中。

这是一个标称注解,只在源代码级有效,主要被编译器用来判断是否真的继承了父类中的方法。

常用的Annotation还有著名的@Deprecated,过时的建议不用的方法。
它的定义如下:

@Documented
@Retention(RetentionPolicy.RUNTIME)
public @interface Deprecated {
}

另外还有通知编译器不要做警告的@SuppressWarnings

@Target( { ElementType.TYPE, ElementType.FIELD, ElementType.METHOD,
        ElementType.PARAMETER, ElementType.CONSTRUCTOR,
        ElementType.LOCAL_VARIABLE })
@Retention(RetentionPolicy.SOURCE)
public @interface SuppressWarnings {

    /**
     * The list of warnings a compiler should not issue.
     */
    public String[] value();
}

元注解

除了上面几个常用的注解定义于最基本的java.lang包中,用于实现这几个Annotation的Annotation都实现在java.lang.annotation包中。它们是被用于实现其它注解所用的。

元素类型 - ElementType

就是这个注解可以用于什么语法单元,比如@Override只能用于方法,方法的类型就是ElementType.METHOD.

这是一个枚举,包括下面的类型:

  • TYPE: 类,接口,枚举
  • FIELD: 域变量
  • METHOD:方法
  • PARAMETER:函数参数
  • CONSTRUCTOR:构造函数
  • LOCAL_VARIABLE:局部变量
  • ANNOTATION_TYPE:注解类型
  • PACKAGE:包

@Target

定义了元素类型,就可以通过这些类型来指定注解适用的类型了。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Target {
    ElementType[] value();
}

@Target注解就是一个ElementType的数组,就像上面我们看到的用法:

@Target( { ElementType.TYPE, ElementType.FIELD, ElementType.METHOD,
        ElementType.PARAMETER, ElementType.CONSTRUCTOR,
        ElementType.LOCAL_VARIABLE })

@Documented

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Documented {
}

用于描述一个注解是可以生成JavaDoc的,也暗示了这个注解是一个公开的API。

@Retention

这是4个元注解中最重要的一个,用于定义这个注解的生命周期。

取值是另一个枚举:

  • RetentionPolicy.SOURCE:注解只存在于源代码中,编译器生成class代码时就忽略了
  • RetentionPolicy.CLASS:会编译进class文件,但是VM执行时调不到
  • RetentionPolicy.RUNTIME:在运行时也可以访问到

RetentionPolicy的定义如下:

27public enum RetentionPolicy {
28    /**
29     * Annotation is only available in the source code.
30     */
31    SOURCE,
32    /**
33     * Annotation is available in the source code and in the class file, but not
34     * at runtime. This is the default policy.
35     */
36    CLASS,
37    /**
38     * Annotation is available in the source code, the class file and is
39     * available at runtime.
40     */
41    RUNTIME
42}

有了这些基础,再看@Retention的实现,就可以完全看得懂了:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Retention {
    RetentionPolicy value();
}

@Retention本身是个运行时可用的注解,公开的API,只对注解本身有效。
它只定义了一个RetentionPolicy枚举的值。

通过反射处理注解

所有的@Target可用的对象都支持用getAnnotations()方法去读取注解。
例如,读取一个类的注解:

        Class clazz = ThreadSafeCounter.class;
        Annotation[] as = clazz.getAnnotations();

我们通过一个例子来说明:

首先先定义一个注解,这个注解用于说明这类或者方法是线程安全的,有一个value用于保存锁对象的名字。

import java.lang.annotation.*;

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.METHOD})
public @interface ThreadSafe {
    String value();
}

下面定义一个使用该注解的类, 类和其中的一个方法都使用这个注解。其实有点废话,类都线程安全了,方法还能不安全么,呵呵

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;

@ThreadSafe("ThreadSafeCounter")
public class ThreadSafeCounter {
    private int mCounter;

    public ThreadSafeCounter(int counter) {
        mCounter = counter;
    }

    @ThreadSafe("this")
    public int incAndGet() {
        synchronized (this) {
            return mCounter++;
        }
    }

下面定义一个main方法去通过反射读注解,先读类的注解:

    public static void main(String[] args){
        Class clazz = ThreadSafeCounter.class;
        Annotation[] as = clazz.getAnnotations();

        for(Annotation a:as){
            ThreadSafe t= (ThreadSafe)a;
            System.out.println("Annotation type="+clazz.getName());
            System.out.println("lock name is:"+t.value());
        }

然后再读取

        Method[] methods = clazz.getMethods();
        for(Method method: methods){
            boolean hasAnno = method.isAnnotationPresent(ThreadSafe.class);
            if(hasAnno){
                ThreadSafe anno = method.getAnnotation(ThreadSafe.class);
                System.out.println("method name="+method.getName()+",lock object="+anno.value());
            }
        }
    }
}

本章内容参考文献:《Java程序设计完全手册》,王作启,伍正云著,北京:清华大学出版社,2014

Android中的Annotation

Android标准的Annotation

@Nullable

定义:

@Retention(SOURCE)
@Target({METHOD, PARAMETER, FIELD})
public @interface Nullable {
}

源码级的,可以用于方法、参数和域,表示一个方法或域的值可以合法地为空,或者是函数的返回值可以合法为空。代码中已经针对为空的情况做了相应的处理。
这是个标称注解。

@NonNull

@Retention(SOURCE)
@Target({METHOD, PARAMETER, FIELD})
public @interface NonNull {
}

与@Nullable相反,@NonNull要求一定不能为空。

@UiThread

定义:

@Retention(SOURCE)
@Target({METHOD,CONSTRUCTOR,TYPE})
public @interface UiThread {
}

表示标有该注解的方法或构造函数应该只在UI线程调用。
如果注解元素是一个类,说明该类的所有方法都应该在UI线程中调用。

@MainThread

@Retention(SOURCE)
@Target({METHOD,CONSTRUCTOR,TYPE})
public @interface MainThread {
}

这个是要求运行在主线程的

@WorkerThread

@Retention(SOURCE)
@Target({METHOD,CONSTRUCTOR,TYPE})
public @interface WorkerThread {
}

要求运行在工作线程

@IntRef

用于定义整数值。

@Retention(CLASS)
@Target({ANNOTATION_TYPE})
public @interface IntDef {
    /** Defines the allowed constants for this element */
    long[] value() default {};

    /** Defines whether the constants can be used as a flag, or just as an enum (the default) */
    boolean flag() default false;
}

我们看一个使用@IntRef例子:

@IntDef({HORIZONTAL, VERTICAL})
@Retention(RetentionPolicy.SOURCE)
public @interface OrientationMode {}

public static final int HORIZONTAL = 0;
public static final int VERTICAL = 1;

在上面定义的@OrientationMode注释中,可以支持的值是HORIZONTAL, VERTICAL.

然后我们再看一个使用flag的例子:

74    @IntDef(flag = true,
75            value = {
76                SHOW_DIVIDER_NONE,
77                SHOW_DIVIDER_BEGINNING,
78                SHOW_DIVIDER_MIDDLE,
79                SHOW_DIVIDER_END
80            })
81    @Retention(RetentionPolicy.SOURCE)
82    public @interface DividerMode {}

View相关的

@RemoteView

支持RemoteView机制,这是一个运行时的注释.

定义路径:/frameworks/base/core/java/android/widget/RemoteViews.java

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

推荐阅读更多精彩内容

  • 本文章涉及代码已放到github上annotation-study 1.Annotation为何而来 What:A...
    zlcook阅读 28,923评论 15 116
  • 什么是注解(Annotation):Annotation(注解)就是Java提供了一种元程序中的元素关联任何信息和...
    九尾喵的薛定谔阅读 3,073评论 0 2
  • 欢迎转载,但请保留作者链接:http://www.jianshu.com/p/82093e5160ae阅读前提:了...
    郭非文阅读 3,451评论 1 14
  • 前言 在Android开发作业中接触到了很多开源框架使用了Java Annotation机制,我接触到的就有Gre...
    Mars_M阅读 735评论 0 2
  • 没有谁是一座孤岛…… “一个人无法自成孤岛,要么至少,一个人无法成为最理想的孤岛。” ——摘自《岛上书店》 如前...
    空知著阅读 304评论 4 1