Java必知必会之(三)--注解

引子:全中国的Java程序员都知道,现在主流的项目架构都是ssm、ssh、springcloud等,而这些框架都离不开spring,而spring中使用了大量的注解(包括spring自定义的注解)。因此想要会用注解,我们就得知道Java注解的原理和基本用法,这样有助于我们在项目中如鱼得水。

在JDK5.0中,新增了很多对现在影响很大的特性,如:枚举、自动装箱和拆箱、注解、泛型等等。其中注解的引入,是为了增加对元数据的支持。

注解:是一个接口,程序可以通过反射来获取指定程序元素的Annotation对象,然后通过Annotation对象来取得注解里的元数据,注解能用来为程序元素(包、类、方法、成员变量等)设置元数据,它不影响程序代码的执行。如果希望让程序中的Annotation在运行时起一定作用,只有通过某种配套的工具对Annotation中的信息进行访问和处理,这个工具同城为APT(Annotation program Tool)

JDK5.0除了增加注解特性之外,还提供了5个基本的Annotation:

  1. @Overrride:限定重写父类方法(旨在强制性提醒)
  1. @Deprecated:表示某个程序元素(类、方法)已过时
  1. @SuppressWarnings:抑制编译警告
  1. @SafeVarargs:堆污染警告
  1. @FunctionalInterface:指定某个接口必须为函数式接口
    注:函数式接口是指一个接口中只包含一个抽象方法(可以包含多个默认方法或多个static方法)

以上这几个注解,在我们的项目经常可见,但是因为他们对程序只是一个强制提醒或者警告作用,并不影响程序的执行,因为我们都没在意这些注解的作用,而当spring出来之后,大量的注解眼花缭乱,作用各异,但他们都有一个共同作用:
让我们在编码过程中简化了不少重复性的代码。
因此我们在了解了注解的原理之后,必须要能自定义一些注解并且使用它到项目中去,才能让我们更好的了解和使用它。
而要想自定义注解,
就必须得了解Java提供的几个元注解
那什么是元注解呢?
元注解:就是负责注解其它注解的注解

在Java5之后定义了4个标准的元注解,分别是:
 1. @Target
 2. @Retention
 3. @Documented
 4. @Inherited

@Target

target注解用来标识注解所修饰的对象范围。

它的可用范围(ElementType的取值范围)有:

  1. CONSTRUCTOR:用于描述构造器(构造方法)
  1. FIELD:用于描述域(成员变量)
  1. LOCAL_VARIABLE:用于描述局部变量(局部变量)
  1. METHOD:用于描述方法(普通方法)
  1. PACKAGE:用于描述包(包定义)
  1. PARAMETER:用于描述参数(如catch等参数)
  1. TYPE:用于描述类、接口(包括注解类型) 或enum声明
  1. ANNOTATION_TYPE:用于注解

使用实例:

1@Target(ElementType.FIELD)2public @interface TargetTest6{3}4@Target({ElementType.TYPE_PARAMETER,ElementType.METHOD})5public @interface TargetTest7{6}

@Retention

@Retention定义了该Annotation被保留的时间长短:某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。使用这个meta-Annotation可以对 Annotation的“生命周期”限制。

它的取值(RetentionPoicy)有:

  1. SOURCE:在源文件中有效(即源文件保留),注解只保留在源代码中,编译器直接丢弃这种注解。
  1. CLASS:在class文件中有效(即class保留),编译器把注解记录在class文件中,当Java程序运行时,JVM不能获取该注解的信息。
  1. RUNTIME:在运行时有效(即运行时保留),编译器将把注解记录在class文件中,当Java运行时,JVM可以获取注解的信息,程序可以通过反射获取该注解的信息。
1@Target(ElementType.FIELD)2@Retention(RetentionPolicy.RUNTIME)3public @interface TargetTest6{4}

@Inherited

@Inherited注解指定被它休市的注解将具备继承性:如果莫个类使用了@XXX注解,则其子类自动被@XXX修饰

1@Target(ElementType.FIELD)2@Retention(RetentionPolicy.RUNTIME)3@Inherited4@interface TargetTest6{5}6//所有使用了@TargetTest6注解的类讲具备继承性,7//也就是它的子类自动带上@TargetTest6注解

@Documented

@Documented用于指定被该注解修饰的注解类将被javadoc工具提取城文档.

1@Target(ElementType.FIELD)2@Retention(RetentionPolicy.RUNTIME)3@Inherited4@Documented5@interface TargetTest6{6}7//javadoc工具生成的API文档将提取@Documented的使用信息

以上就是所有的元注解以及他们的作用分析,
有了这些元注解有什么用呢?
。。。。
当然是为了方便我们在项目中自定义注解咯
那自定义注解怎么自定义呢?
下面我们来看看介绍如何自定义注解并利用注解完成一些实际的功能


语法:

1类修饰符 @interface 注解名称{2 //成员变量,在注解中以无形参的形式存在3 //其方法名和返回值定义了该成员变的名字和类型4 String name();5 int age(); 6}

可以看到:注解的语法和接口的定义非常类似,他也一样具备有作用域,但是它的成员变量的定义是以无形参的方法形式存在,名字定义了成员变量的名字,返回值定义了变量类型。
下面我们来自定义一个注解:

1@Target(ElementType.FIELD)2@Retention(RetentionPolicy.RUNTIME)3public @interface AnonTest {4    int age();5    String name();6}

注解@AnonTest在运行时有效,作用域在成员变量上,它有两个成员变量分别是int型的age和String型的name。
接下来我们就可以使用这个注解了

1public class test {    2     @AnonTest(age = 0, name = "1")3     public Integer tes;4}

这样就是一个注解的定义和使用了,有人会疑惑说,spring中很多注解都是@xxx,为什么这个@AnonTest一定要要带上两个成员变量呢?
原因很简单:
注解中的成员变量如果没有默认值,则在使用注解时必须要给成员变量赋值
但如果成员变量有默认值,那可以直接在定义注解时,赋值上去,这样在使用时就可以省略不写

1@Target(ElementType.FIELD)2@Inherited3@Retention(RetentionPolicy.RUNTIME)4public @interface AnonTest {5    int age() default 1;6    String name() default "注解辣鸡";7}

这样在调用注解时就可以不赋值

1public class test {23    @AnonTest4    public Integer tes;

上面看到,我们已经使用了注解,但是我们并没发现@AnonTest对我们的tes成员变量有任何作用,这是因为注解本身在程序中是不会生效的,而是需要程序来提取数据并且处理注解本应该做的工作。
因此
我们需要知道如何从注解中提取信息并且做相应的处理。
Java使用Annotation接口来代表程序元素前面的注解,该接口是所有注解的父接口,该接口主要有以下几个实现类:

  1. Class:类定义
  1. Constructor:构造器定义
  1. Field:类成员变量定义
  1. Method:类的方法定义
  1. Package:类的包定义

AnnotatedElement接口是所有程序元素所实现的接口的父接口,所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下几个方法来访问注解信息:

  1. getAnnotation(Class

    <t style="margin: 0px; padding: 0px; max-width: 100%; box-sizing: border-box !important; word-wrap: break-word !important; font-size: inherit; color: inherit; line-height: inherit;">annotationClass): 返回改程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。</t>

  2. getAnnotations():返回该程序元素上存在的所有注解。

  1. isAnnotationPresent(Class annotationClass):判断该程序元素上是否包含指定类型的注解,存在则返回true,否则返回false.
  1. getDeclaredAnnotations():返回直接存在于此元素上的所有注释。

有了这些方法,我们就可以在类、方法、成员变量等程序元素中获取到注解信息
比如:

 1@Component 2@Service 3public class test { 4 5    @AnonTest 6    public Integer tes; 7 8    @Deprecated 9    public void m1(){1011    }1213    public static void main(String[] args) {14        try {15            //获取所有test类上的注解16            Annotation[] ar=Class.forName("com.anon.test").getAnnotations();17            //获取所有test类上m1方法的注解18            Annotation[] ar1=Class.forName("com.anon.test").getMethod("m1").getAnnotations();19            //遍历所有的注解,检测是否有我们想要的某个注解 如@Component注解20            for(Annotation a:ar){21                System.out.println(a.getClass());22                if(a instanceof  Component)23                System.out.println("用了注解:"+a);24            }25        } catch (Exception e) {26            // TODO Auto-generated catch block27            e.printStackTrace();28        }29    }30}3132//输出:33//用了注解:@org.springframework.stereotype.Component(value=)34//class com.sun.proxy.$Proxy4

利用AnnotatedElement提供的方法能获取到我们想要的注解信息之后,就可以针对注解做一些特定的行为。

例如,对于所有的注册用户信息,系统需要把名称和年龄上报到另一个年龄分布表中,就可以利用注解就可以完成这样的动作

 1public class test { 2    @AnonTest(name="张小龙",age=25) 3    public Integer tes; 4    public static void main(String[] args) { 5        test.getInfo(test.class); 6    } 7    public static void getInfo(Class<?> clazz) { 8        //获取目标类的所有成员变量 9        Field[] fields = clazz.getDeclaredFields();10        for (Field field : fields) {11            //查找变量中是否有存在@AnonTest注解12            if (field.isAnnotationPresent(AnonTest.class)) {13                //如果存在,则做相应处理 14                AnonTest anon= field.getAnnotation(AnonTest.class);15                 System.out.println("名称是:"+anon.name());16                 System.out.println("年龄是:"+anon.age());17                 System.out.println("上报信息到用户年龄分布表中");18                 //上报信息到用户年龄分布表中19                 //uploadInfoToLog(anon.name()),anon.age())20            }21        }22    }23}

输出:

1名称是:张小龙2年龄是:253上报信息到用户年龄分布表中

最终程序按照我们的需求运行并且输出正确的信息,由此可见,注解对于Java来说是一种补充,他的存在与否对程序来说应该是无影响的,灵活使用注解能使开发的效率更高,而且程序规范性也得到增强。

强烈推荐

1Redis模糊查询在生产环境出现严重性能问题

2Linux性能的重要指标:打开文件数的限制

3以Vue为例,解释JavaScript的反应性

4Java必知必会之----Enum枚举类揭秘

518位身份证号藏什么玄机?用js教你校验


觉得本文对你有帮助?请分享给更多人

关注「编程无界」,提升装逼技能

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

推荐阅读更多精彩内容

  • 01.是否有堆积了很久没穿过的衣服,因为优惠买的不常用的小物件,像这些的话我们都可以整理出来,然后处理掉。 02....
    九悠阅读 138评论 0 3
  • 姓名:母光艳 公司:宁波贞观电器 【日精进打卡第12天】 【知-学习】 背诵《六项精进》1遍,共56遍, 诵读《大...
    母光焱阅读 115评论 0 1