注解(Annotation)在java中应用非常广泛。它既能帮助我们在编码中减少错误,(比如最常见的Override注解),还可以帮助我们减少各种xml文件的配置,比如定义AOP切面用@AspectJ模式代替Schema模式,特别是最近接触了一点Spring MVC,每次编写Controller的时候用@RequestMapping(),@RequestParam,@ResponseBody等等。

我们所用的java中自带的注解都是定义好了的,直接拿来用,它就能发挥作用。当然了,自己也可以定义注解。

注解定义:

public @interface 注解名 {

    //…
}

可以给注解中可以定义“属性”,暂且理解为属性吧,定义如下:

public @interface MyAnnotation {

    String hello(); // 定义了一个属性名hello

    String world() default "world";// 定义了一个属性名world,默认值为“world”,注意类型匹配

    int number();// 定义了一个属性名为number

    String[] value(); // 定义了一个属性为value的数组

}

后面必须有括号,修饰符只能为public或abstract,不过一般都是默认不写的。

当我们定义一个注解时,它暗自的继承了Annotation这个借口。如果手动定义了一个Interface 并继承了java.lang.annotation.Annotation,并不意味着定义了一个注解。注解只能通过@Interface方式定义。Annotation是一个接口而不是一个注解。

使用方式:

1.当我们对一个注解定义了多个属性后,在使用该注解时,除了定义了默认值的属性,其他属性都要赋值。当然,也可以显式的对含有默认值的属性赋值。

例如:

public class MyAnnotationTest{

    private String name;
@MyAnnotation(hello = "", number = 0, value = { "aa" },world="world1")
public static void main(String[] args) {
MyAnnotationTest test = new MyAnnotationTest();
test.method();
}
@MyAnnotation(hello = "hello", number = 0, value = {"a"},world="aa")
public void method(){
System.out.println("method");
}
}

2.当一个注解只含有一个属性且属性名为value时,我们可以简化使用方式。

比如定义一个注解:
public @interface SingleAnnotation {
String[] value();
}

在使用在注解时可以简化:

    @SingleAnnotation("value")
public void singleAnnotationMethod(){ }

注解基本的定义和使用差不多就这些了。接下来看看定义的注解为什么会生效。

@Retention注解

为了更好的使用注解,java内置的这个注解也是需要了解的。

java.lang.annotation.Retention可以告诉编译器怎样去对待和处理我们自定义的注解。默认情况下会将Annotation的信息留在class文件中,但不会被JVM读取。

在使用Retention注解时,需要提供java.lang.annotation.RetentionPolicy的枚举值。编译器会根据设置不同的值来决定如何处理定义的注解。

RetentionPolicy定义如下:

public enum RetentionPolicy {

    SOURCE, //编译程序丢弃该注解,表示他不会存在于class文件中

    CLASS, //编译程序将Annotation存在class文件中(缺省)

    RUNTIME //编译程序将Annotation存在class文件中,可由JVM读取
}

比如@SuppressWarnings注解,它所对应的@Retention(RetentionPolicy.SOURCE)是这样设置的,因为它的作用是告知编译程序来压制警告,没必要将这个信息存在于class文件中。

RetentionPolicy.RUNTIME:搭配反射机制,可以让注解被JVM读出。

一个注解信息如何被JVM读出

  1.将该注解加上 @Retention(value = RetentionPolicy.RUNTIME)。

  1.获得一个类的class对象

  2.获得这个类中被该注解修饰的方法或属性。 AccessibleObject的isAnnotationPresent(Class<? extends Annotation> annotationClass)方法可以判断一个注解是否出现,由于Field和Method类继承了AccessibleObject,所以可以判断出一个类的方法或属性是否有该注解。又因为Method实现了AnnotatedElement ,所以method可以得到它对应的注解的Annotation对象,进而可以获取注解的相关信息。

下面给出例子:

首先给MyAnnotation注解 加上 @Retention(value = RetentionPolicy.RUNTIME)。

public class MyAnnotationTest{

    public static void main(String[] args)throws Exception{
MyAnnotationTest test = new MyAnnotationTest(); Class<?> clazz = test.getClass();
Method[] methods = clazz.getMethods(); for(Method m : methods){
//该方法是否出现MyAnnotation注解,出现为true,否则为false
if(m.isAnnotationPresent(MyAnnotation.class)){ MyAnnotation annotation = m.getAnnotation(MyAnnotation.class);
String hello = annotation.hello();
int number = annotation.number();
String[] values = annotation.value();
String world = annotation.world(); m.invoke(test,new Object[]{}); System.out.println("annotation.hello():" + hello);
System.out.println("annotation.number():" + number);
System.out.println("annotation.value():" + values[0]);
System.out.println("annotation.world():" + world); }
}
} @MyAnnotation(hello = "hello", number = 100, value = {"a"},world="aa")
public void method(){
System.out.println("method");
}
public void method2(){
System.out.println("method2");
} }

console 结果:

结果一目了然,可以让该方法执行,可以拿到该注解的所有赋值情况。

如果 MyAnnotation的Retention注解为 @Retention(value = RetentionPolicy.CLASS) 或@Retention(value = RetentionPolicy.SOURCE),那么method方法也不会执行,更不会获取到MyAnnotation的先关信息了。

@Target注解

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

它也是用来注解我们定义的注解。value为ElementType类型的枚举值。

ElementType定义:

public enum ElementType {
/** Class, interface (including annotation type), or enum declaration */
TYPE, /** Field declaration (includes enum constants) */
FIELD, /** Method declaration */
METHOD, /** Parameter declaration */
PARAMETER, /** Constructor declaration */
CONSTRUCTOR, /** Local variable declaration */
LOCAL_VARIABLE, /** Annotation type declaration */
ANNOTATION_TYPE, /** Package declaration */
PACKAGE
}

好吧,不用解释了,看注释都知道每一个值是什么意思了。

如果一个注解被@Target直接,那么该注解所修饰的类型将受到限制,如果修饰的类型不匹配编译将会报错。

表示MyAnnotation注解只能修饰方法,可根据ElementType 的值以此类推。

@Documented

它表示被修饰的注解将会生成到javadoc帮助文档中。

@Inherited:允许子类继承父类的注解。缺省情况下子类并不会继承父类中的Annotation.

就算java的原生注解,一般也是被这几种注解中的一种或多种修饰,进而达到其功能。

总结:个其实java注解的原理还是反射。在运行时,因为有了注解,利用反射机制在运行时才会找到注解对应的方法,找到注解对应的值。值拿到了,方法也找到了,想干什么就干什么了,该干什么这一般应该是底层或框架做的事情。但我想应该就是这么回事吧!

java注解(Annotation)解析的更多相关文章

  1. Java注解全面解析(转)

    1.基本语法 注解定义看起来很像接口的定义.事实上,与其他任何接口一样,注解也将会编译成class文件. @Target(ElementType.Method) @Retention(Retentio ...

  2. Java注解(Annotation)详解

    转: Java注解(Annotation)详解 幻海流心 2018.05.23 15:20 字数 1775 阅读 380评论 0喜欢 1 Java注解(Annotation)详解 1.Annotati ...

  3. Java注解Annotation与自定义注解详解

    Java注解简介 开发中经常使用到注解,在项目中也偶尔会见到过自定义注解,今天就来探讨一下这个注解是什么鬼,以及注解的应用场景和如何自定义注解. 下面列举开发中常见的注解 @Override:用于标识 ...

  4. 深入JAVA注解-Annotation(学习过程)

    JAVA注解-Annotation学习 本文目的:项目开发过程中遇到自定义注解,想要弄清楚其原理,但是自己的基础知识不足以支撑自己去探索此问题,所以先记录问题,然后补充基础知识,然后解决其问题.记录此 ...

  5. 深入学习JAVA注解-Annotation(学习过程)

    JAVA注解-Annotation学习 本文目的:项目开发过程中遇到自定义注解,想要弄清楚其原理,但是自己的基础知识不足以支撑自己去探索此问题,所以先记录问题,然后补充基础知识,然后解决其问题.记录此 ...

  6. Java - 注解 (Annotation)

    Java - 注解 (Annotation)   一.基本的 Annotation     > 使用 Annotation 时要在其前面增加 @符号,并把该 Annotation 当成一个修饰符 ...

  7. Java注解Annotation(一)

    Java注解Annotation(一)——简介 这一章首先简单介绍一下注解,下一章会给出一个注解应用的DEMO. 1. 元注解 元注解的作用是负责注解其他的注解. JDK1.5中,定义了4个标准的me ...

  8. Java注解Annotation学习

    学习注解Annotation的原理,这篇讲的不错:http://blog.csdn.net/lylwo317/article/details/52163304 先自定义一个运行时注解 @Target( ...

  9. java注解Annotation

    扯扯注解的蛋 为什么学习注解?学习注解有什么好处?学完能做什么? 1.能够读懂别人的代码,特别是框架相关的代码 2.让编程更加简洁,代码更加清晰 3.让别人高看你一眼 注解是java1.5引入的 概念 ...

随机推荐

  1. 转载:bootstrap, boosting, bagging 几种方法的联系

    转:http://blog.csdn.net/jlei_apple/article/details/8168856 这两天在看关于boosting算法时,看到一篇不错的文章讲bootstrap, ja ...

  2. spring mvc 参数

    Struts(表示层)+Spring(业务层)+Hibernate(持久层) Struts: Struts是一个表示层框架,主要作用是界面展示,接收请求,分发请求. 在MVC框架中,Struts属于V ...

  3. VR定制 AR定制 就找北京动软VR开发团队(VR案例 AR案例)

    我们长期承接丰交互软件.游戏项目外包: VR/AR内容应用定制.VR.AR游戏项目外包(有主流测试硬件设备) VR全景应用.视频外包 请提供贵公司的信息,我们将提供高大上的VR案例欢迎联系我们给您提供 ...

  4. Adaptive Decontamination of the Training Set: A Unified Formulation for Discriminative Visual Tracking

    Martin Danelljan 判决类追踪模型是由训练样本学习得到,但是为了适应目标和背景的变化sample set在每一帧中都会更新. 令(xjk, yjk)表示第k帧k={1,2,...,t}中 ...

  5. Eclipse下的Maven

    本文转载自:http://www.cnblogs.com/zlslch/p/5882567.html 当我们无法从本地仓库找到需要的构件的时候,就会从远程仓库下载构件至本地仓库.一般地,对于每个人来说 ...

  6. Struts1.x有两个execute方法,不要重写错哦HttpServletRequest才是对的(转)

    Struts1.x 的 Action 有两个 execute 哦,小心搞错! by agate - Published: 2008-05-01 [9:42 下午] - Category: 程序编码 不 ...

  7. 微信开发-ACCESS TOKEN 过期失效解决方案

    微信开发-ACCESS TOKEN 过期失效解决方案 起因:因为access_token的重要性,开发过微信的都知道,但是他有自己的生命周期,官方解释为:"有效期为7200秒",一 ...

  8. git基本用法说明(原创+验证)

      关于文件状态   一般仓库中的文件可能存在于这4种状态: 1)Untracked files                     → 文件未被跟踪(A)  2)Untracked but no ...

  9. R语言-数据结构

    1.向量 向量是用来存储数值型.字符型或逻辑性数据的一维数组,用函数c()创建向量 a <- c(1,2,5,6,4) b <- c("one","two&q ...

  10. VS2010中App_Code文件夹的问题

    在VS2010中新建一个Web Application,然后新建一个app_Code文件夹, 在app_code文件夹下建一个ClassHelper类. 然后在index页面中使用ClassHelpe ...