Java注解入门

示例代码github地址

注解的定义

在Java中,要声明一个注解很简单,只要你会使用接口,你就一定会使用注解,以下是一个简单的注解的例子:

public @interface Table{
    // add code to yourself
}

聪明的你一定会发现,把接口变成注解只需要再interface前增加一个@符号就可以了.

Java中的元注解

@Target

Target注解声明了Annotation所修饰对象的范围,它可携带的参数可枚举如下:

public enum ElementType {    
/**类|接口|枚举类型
*Class, interface (including annotation type), or enum declaration */   
TYPE,  
/**成员变量 包括枚举常量
*Field declaration (includes enum constants)*/    
FIELD,   
/** 方法声明 Method declaration  */   
METHOD,   
/** 形参声明 Formal parameter declaration */  
PARAMETER,  
/** 构造函数 Constructor declaration */ 
CONSTRUCTOR,  
/** 局部变量 Local variable declaration */   
LOCAL_VARIABLE,  
/**  Annotation type declaration */   
ANNOTATION_TYPE,  
/** 包 Package declaration */    
PACKAGE,   
/**     
  * Type parameter declaration     *    
  * @since 1.8    java 1.8 以上新增支持
  * @hide 1.8    
 */   
 TYPE_PARAMETER,   
 /**  
 * Use of a type
 * @since 1.8     java 1.8 以上新增支持  
 * @hide 1.8     
 */  
  TYPE_USE
}

@Retention

该注解主要是声明Annotation被保留的时间线,有些Annotation值保留在源码中,有些Annotation 会被编译到class文件中,而有些Annotation会被JVM所读取,这些都是@Retention在起控制作用。

public enum RetentionPolicy {   
/**存在于源码中,注解将被编译器丢弃*/ 
SOURCE,    
/** 注解会被编译到CLASS文件中,注解将被编译器记录到类文件中,但在运行时将被VM所丢弃/  
CLASS,    
/**它会一直保留到VM运行时,因此它可以被反射机制所捕获*/    
RUNTIME
}

@Documented

需要被javadoc所记录

@Inherited

该注解 声明了某个被标注的类型是被继承,如果一个使用Inherited的注解被用于<b>Class</b>中,那么它的子类也会继承这个注解.

解析注解

使用以下方法可以访问Annotation的注解信息

/**注解属性是否是指定类型*/
public boolean isAnnotationPresent(Class<? extends Annotation> annotationType)
/**返回指定类型的注解*/
public <A extends Annotation> A getAnnotation(Class<A> annotationType)
/**返回所有的注解(该方法将忽略继承到的注解类型)*/
public native Annotation[] getDeclaredAnnotations()
/**返回所有的注解(该方法将包含继承到的注解类型)*/
public Annotation[] getAnnotations()

代码实现

package com.deity.application.annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 自定义注解
 * Created by Deity on 2017/2/8.
 */

@SuppressWarnings("unused")
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface HumanAnnotation {
    /**访问权修饰符 public 或者 protect*/
    String userName() default "还未起名字";
    String getMyBrithday() default "2016-09-10 10:00:00";
    int getMyAge() default 1;
}

package com.deity.application.annotation;

/**
 * 女儿
 * Created by Deity on 2017/2/8.
 */
public class Daughter {
    /**起个响亮的名字*/
    private String userName;
    /**生日*/
    private String birthday;
    /**年龄*/
    private int age;

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getBirthday() {
        return birthday;
    }

    public void setBirthday(String birthday) {
        this.birthday = birthday;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    @HumanAnnotation(userName = "小宝宝",getMyBrithday = "2016年2月8日22:11:50",getMyAge = 2)
    public String toString() {
        String result = "这是我女儿>>>姓名:"+userName+" 年龄:"+age+" 出生日期:"+birthday;
        return result;
    }
}

package com.deity.application;

import com.deity.application.annotation.Daughter;
import com.deity.application.annotation.HumanAnnotation;

import org.junit.Test;

import java.lang.reflect.Method;

/***
 * 自定义注解测试
 */
public class ExampleUnitTest {

    @Test
    public void AnnotationTest(){
        Daughter daughter = new Daughter();
        Method[] fields = Daughter.class.getDeclaredMethods();
        for (Method field:fields){
            if (field.isAnnotationPresent(HumanAnnotation.class)) {
                HumanAnnotation humanAnnotation = field.getAnnotation(HumanAnnotation.class);
                daughter.setAge(humanAnnotation.getMyAge());
                daughter.setBirthday(humanAnnotation.getMyBrithday());
                daughter.setUserName(humanAnnotation.userName());
            }
        }
        System.out.println(daughter.toString());
    }
}
执行结果

实现一个编译时注解

package com.example;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 注解Demo
 * Created by Deity on 2017/2/8.
 */

@SuppressWarnings("unused")
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface HumanMessageAnnotation {
    /**public Or protect*/
    String userName() default "unknow";
    String getMyBrithday() default "2016-09-10 10:00:00";
    int getMyAge() default 1;
}

注解器实现:

package com.example;

import com.google.auto.service.AutoService;

import java.util.LinkedHashSet;
import java.util.Set;

import javax.annotation.processing.AbstractProcessor;
import javax.annotation.processing.ProcessingEnvironment;
import javax.annotation.processing.RoundEnvironment;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.TypeElement;
import javax.annotation.processing.Processor;

/**
 * 1.继承 AbstractProcessor
 */

//@SupportedSourceVersion(SourceVersion.RELEASE_6)  1
//@SupportedAnnotationTypes("com.example.HumanMessageAnnotation") 2
@AutoService(Processor.class)
public class MyProcessor extends AbstractProcessor{

    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
    }

    /**
     * {@inheritDoc}
     *
     * @param annotations
     * @param roundEnv
     * @return true
     */
    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        for (Element element : roundEnv.getElementsAnnotatedWith(HumanMessageAnnotation.class)) {
            if (ElementKind.METHOD == element.getKind()){
                ExecutableElement typeElement = (ExecutableElement ) element;
                System.out.println(typeElement.getAnnotation(HumanMessageAnnotation.class).userName()+"\n"+
                        typeElement.getAnnotation(HumanMessageAnnotation.class).getMyBrithday()+"\n"+
                        typeElement.getAnnotation(HumanMessageAnnotation.class).getMyAge());
            }
        }
        return false;
//        for (Element element : roundEnv.getElementsAnnotatedWith(MyAnnotation.class)) {
//            System.out.println("------------------------------");
//            if (element.getKind() == ElementKind.CLASS) {
//                TypeElement typeElement = (TypeElement) element;
//                System.out.println(typeElement.getSimpleName());
//                System.out.println(typeElement.getAnnotation(MyAnnotation.class).value());
//            }
//            System.out.println("------------------------------");
//        }
//        return false;
    }


    @Override
    public Set<String> getSupportedAnnotationTypes() {
        Set<String> annotations = new LinkedHashSet<>();
        annotations.add(HumanMessageAnnotation.class.getCanonicalName());//基础类的规范名称
        return annotations;

//        Set<String> annotataions = new LinkedHashSet<String>();
//        annotataions.add(MyAnnotation.class.getCanonicalName());
//        return annotataions;
    }


    @Override
    public SourceVersion getSupportedSourceVersion() {
        return SourceVersion.latestSupported();
    }
}

扩展
javax.lang.model.element 中 Element 的子接口
................................................................................................................
interface ExecutableElement
表示某个类或接口的方法、构造方法或初始化程序(静态或实例),包括注释类型元素。
interface PackageElement
表示一个包程序元素。
interface TypeElement
表示一个类或接口程序元素。
interface TypeParameterElement
表示一般类、接口、方法或构造方法元素的形式类型参数。
interface VariableElement
表示一个字段、enum常量、方法或构造方法参数、局部变量或异常参数。

注解处理器的注册

手动注册注解处理器

1、在 processors 库的 main 目录下新建 resources 资源文件夹;
2、在 resources文件夹下建立 META-INF/services 目录文件夹;
3、在 META-INF/services 目录文件夹下创建 javax.annotation.processing.Processor 文件;
4、在 javax.annotation.processing.Processor 文件写入注解处理器的全称,包括包路径;

使用AutoService 实现注解处理器的注册

compile 'com.google.auto.service:auto-service:1.0-rc3'

使用APT

在工程build.gradle上,增加依赖

classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'  // add

在module上增加依赖

apply plugin: 'com.neenbedankt.android-apt' // add 

dependencies {  
      //.............
     compile project(':annotations')
     apt project(':processor') 
}
注解的使用

运行结果

参考:
深入理解Java:注解Annotation自定义注解入门

APT(Annotation Processing Tool) 原理
APT 是一种处理注释的工具,对Android开发者来说,市面上比较出名的就是android-apt,但是android-apt,从Android Studio 2.2开始,Google已经内置了一款APT插件,那就是annotationProcessor,目前android-apt已宣布android-apt完成它的历史使命,并推荐大家使用官方annotationProcessor插件.

扩展阅读 https://juejin.im/entry/58fefebf8d6d810058a610de

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

推荐阅读更多精彩内容

  • 本文章涉及代码已放到github上annotation-study 1.Annotation为何而来 What:A...
    zlcook阅读 29,144评论 15 116
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,028评论 25 707
  • 前面写了Android 开发:由模块化到组件化(一),很多小伙伴来问怎么没有Demo啊?之所以没有立刻放demo的...
    涅槃1992阅读 8,022评论 4 37
  • Java注解(Annotation) 0.0 Hello World 先上代码,再加以说明。这样不至于让初学者懵。...
    GAWEGOR阅读 421评论 0 3
  • 现在社会,会做事,更要会做人。做事前,必须得面对相关对象做好宣传工作。宣传什么呢,要做什么事,会遇到什么困难。尤其...
    香杉阅读 310评论 1 1