Java注解

简介

由于无论在Java后台或者Android开发中我们经常遇到注解这个功能机制,例如常用的框架Java后台开发中,Spring、MyBatis等,Android的Dagger2,butterknife等,都是注解框架。今天我们就了解java是如何进行设置注解的?我们可以定义一个注解,方便我们使用等等。

注解元

在进行了解注解时我们先来了解一下,一般注解主要包含以下几个重要的注解元,java注解的机制离不开这几个注解元。

注解元 介绍<功能>
1.@Target 注解用于什么地方,下面会介绍
2.@Retention 什么时候使用该注解
3.@Documented 注解是否将包含在JavaDoc中
4.@Inherited 是否允许子类继承该注解,表示父类如果添加此注解,子类也可以使用
5.@Repeatable java8添加的,可重复的,表该注解可以多次使用

注解的定义

注解的定义很简单,就是通过@interface来进行修饰注解,就是平时我们定义接口的时候前面添加一个@符号。当然了,注解的定义需要添加一些常用的注解元,才能有效的构造我们所需的注解。如下定义注解:

public @interface TestAnnotation {
}

注解元解释

@Target

通过@Target进行添加到注解中,说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了@target可更加明晰其修饰的目标,用于什么地方,修饰什么东西,如下常用的几种方式。

  • CONSTRUCTOR: 用于描述构造器
  • FIELD:用于描述域
  • LOCAL_VARIABLE:用于描述局部变量
  • METHOD:用于描述方法
  • PACKAGE:用于描述包
  • PARAMETER:用于描述参数
  • TYPE:用于描述类、接口(包括注解类型) 或enum声明
    如下代码
@Target({ElementType.FIELD,ElementType.METHOD}) //多个的时候使用{}括起来,然后用逗号分隔开
public @interface TargetTest {
}

@Retention

该注解是表示我们在运行或者编译时的一个保留状态,指示要保留带注释类型的注释的时间长度。

  • SOURCE:在源文件中有效(即源文件保留),在编译时将其抛弃掉。
  • CLASS:在class文件中有效(即class保留),不会添加载到JVM中
  • RUNTIME:在运行时有效(即运行时保留),这个和我们常用的加载是一致的,运行时会加载到JVM中,一般通过反射可获取到,一般这个是我们常用的。
@Retention(RetentionPolicy.RUNTIME)//在使用该注解其,只能选择其中一种属性,不能定义多个
public @interface RetentionTest {
}

@Documented

@Documented用于描述其它类型的annotation应该被作为被标注的程序成员的公共API。
如下代码

@Documented //添加文档注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnotationTest {
}

@Inherited:

@Inherited元注解是一个标记注解, @Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。

@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Inherited  //添加继承注解
public @interface AnnotationTest {
}

@Repeatable

这个是在java8添加的注解特性,@Repeatable的值表示可重复注释类型的包含注释类型。

@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Repeatables {
    RepeatablesTest[] value();
}
@Repeatable(Repeatables.class)
public @interface RepeatablesTest {
    String value() default "哈哈哈";
}

例子

我们简单的介绍了注解的常用的5个注解元,解下来我们通过例子来实现注解。
代码如下,根据上面的提示我们进行测试。
在看例子之前我们先来了解一下常用的几个方法。

  • Class.getAnnotations() 获取所有的注解,包括自己声明的以及继承的
  • Class.getAnnotation(Class< A > annotationClass) 获取指定的注解,该注解可以是自己声明的,也可以是继承的
  • Class.getDeclaredAnnotations()获取自己声明的注解
  • Class.getDeclaredField(String name); //获取该类的声明字段
  • Class.getDeclaredMethods();//返回的是一个Method[]数组
@Documented
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnotationTest {
    String name() default "小明"; //表示默认为小明。
}
@AnnotationTest() //在该类添加注解
public class TestAnnotationMain {

    public static void main(String[] args) {
        boolean hasAnnotation = TestAnnotationMain.class.isAnnotationPresent(AnnotationTest.class);
        if (hasAnnotation) {
            AnnotationTest annotation = TestAnnotationMain.class.getAnnotation(AnnotationTest.class);
            System.out.println(annotation.name());
        }

    }
}

打印输出结果:

输出1.png

如果我们想更改name的值可以这么弄

@AnnotationTest(name = "小刚")
public class TestAnnotationMain {
    public static void main(String[] args) {
        boolean hasAnnotation = TestAnnotationMain.class.isAnnotationPresent(AnnotationTest.class);
        if (hasAnnotation) {
            AnnotationTest annotation = TestAnnotationMain.class.getAnnotation(AnnotationTest.class);
            System.out.println(annotation.name());
        }
    }
}
输出2.png

如果我们想给一个类的属性进行赋值可以这么做
1.创建一个注解如下

@Documented
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnotationTest1 {
    String value(); //value来定义
}

2.引用该主解

public class Test {
    @AnnotationTest1(value = "小云") //引用注解进行赋值
    public String name;
}

3.测试

public class TestAnnotation1Main {
    public static void main(String[] args) {
        try {
            Field name = Test.class.getDeclaredField("name");//该该类的字段
            name.setAccessible(true); 
            AnnotationTest1 annotationTest1 = name.getAnnotation(AnnotationTest1.class);//获取该字段的注解
            if (annotationTest1 != null) {
                System.out.println(annotationTest1.value()); //输出值
            } 
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        }
    }
}

获取方法上的注解类 如AnnotationTest2

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface AnnotationTest2 {
     //todo无任何方法或者属性
}
public class Test {

    @AnnotationTest1(value = "小云")
    public String name;

    @AnnotationTest2 //目的获取该AnnotationTest2 
    public void fun() {
        System.out.println("方法执行");
    }
}

获取

public class TestAnnotation1Main {
    public static void main(String[] args) {
        try {
            Field name = Test.class.getDeclaredField("name"); //获取该类的声明字段
            name.setAccessible(true);
            AnnotationTest1 annotationTest1 = name.getAnnotation(AnnotationTest1.class);//获取该字段的注解
            if (annotationTest1 != null) {
                System.out.println(annotationTest1.value()); //输出值
            }

            Method fun = Test.class.getDeclaredMethod("fun");
            if (fun != null) {
                Annotation[] annotations = fun.getAnnotations();
                for (int i = 0; i < annotations.length; i++) {
                    System.out.println(annotations[i].annotationType().getSimpleName());
                }
            }
        } catch (NoSuchFieldException | NoSuchMethodException e) {
            e.printStackTrace();
        }
    }
}

举例

@Documented
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
}
////////////////////////////////////////////////////////////////////////////
public class MyTest { //进行获取注解方法的全部数据
    @MyAnnotation
    public void mytestLoad() {
        System.out.println("测试加载");
    }
    @MyAnnotation
    public void mytestRequest() {
        System.out.println("测试请求");
    }
    @MyAnnotation
    public void mytestProgress() {
        System.out.println("测试进度");
    }
    @MyAnnotation
    public void mytestError() {
        System.out.println(1 );
    }
   ///////该方法不执行
   public void mytestNoAnno(){
        System.out.println("没有注解的方法");
    }
}
////////////////////////////////////////////////////////////////////////////
public class TestMain {
    public static void main(String[] args) {
        MyTest myTest = new MyTest();
        Method[] methods = myTest.getClass().getDeclaredMethods();
        for (int i = 0; i < methods.length; i++) {
            Method method = methods[i];
            if (method.isAnnotationPresent(MyAnnotation.class)) {
                try {
                    method.setAccessible(true);
                    method.invoke(myTest, null);//调用该类的注解方法
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                } catch (InvocationTargetException e) {
                    e.printStackTrace();
                }
            }
        }
        System.out.println("==================输出完成!====================");
    }
}
输出3.png

下载

Annotation案例代码下载

总结

我们在开发中长见的注解如下:

常用注解 解释
@Override 方法重写
@SuppressWarnings 提示警告
@SafeVarargs 参数安全
@Deprecated 作用是对不应该再使用的方法添加注解
@FunctionalInterface 函数式编程,用于Lambda 表达式

注解给我们带来了许多方便,但是我们也得知道其优缺点。

  • 优点
    注解方便我们进行单元测试,有利于进行开发。
    代码整洁,通过注解的方式变可知修饰的变量,或者方法。
    节省配置,减少配置文件大小。

  • 缺点
    通过反射进行设置,可能会产生性能上的问题。
    若要对配置进行修改需要重新编译,扩展性差。

参考

https://blog.csdn.net/briblue/article/details/73824058

https://blog.csdn.net/ryo1060732496/article/details/80891093

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

推荐阅读更多精彩内容