Annotations 0

java annotations basics

What are Anotations?

如果用一个单词来描述Annotation那就是Metadata, Metadata是描述数据的数据,所以Annotation可以理解为code的元数据描述。Annotation在JDK5.0版本被引入,大概是如下图这样的

@Override
public String toString() {
    return "This is String Representation of current object."
}

这里重载了toString()方法,用到了@Override注解。如果去掉了@Override,代码也能正常运行。那为什么还要用它呢?@Override注解告诉编译器这是一个重载方法,是关于method的描述。如果父类中没有这个方法,那么编译器会报错,如下:

Error:(8, 5) java: 方法不会覆盖或实现超类型的方法

重新定义:
Annotation是用来装饰class、method、field、parameter、variable、constructor和package的一种特殊Java结构体。

How Annotations Work

Java内置的Override注解源码如下:

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

可以看出Override没有什么东西,又怎么做到检验重载方法是否定义在父类中。原因是Annotation只是metadata,不包含任何的业务逻辑。可以这么来理解,Annotation是class、method、package、field的装饰信息的生产者,对应的还有一个信息的消费者。那这里的@Override的问题就好理解了,Override是信息的提供者,JVM是信息的消费者,在二进制层面来检测重载方法的是否存在于父类中。

How to Write Custom Annotations

JDK提供了4中内置的元注解,作用是注解其他的注解,分别是

@Documented - Whether to put the annotation in Javadocs
@Retention - When the annotation is needed
@Target? - Places the annotation can go
@Inherited - Whether subclasses get the annotation

@Retention - Annotation被保留的时间长短,标注注解的生命周期。3种RetentionPolicy的取值说明如下

RetentionPolicy.SOURCE - 在编译期间就被抛弃。这类注解在编译完成后就没有任何意义了,不会被写到字节码里。比如@Override、@SuppressWarnings

RetentionPolicy.CLASS - 在class加载时被抛弃。适合在编译成字节码后做一些处理,而且是Retention的默认值。

RetentionPolicy.RUNTIME - 不抛弃。这类注解适用于运行时反射

@Target - 注解放置的位置
如果不指定@Target的话,可以放到任何地方,一般的使用如下

ElementType.TYPE (class, interface, enum)
ElementType.FIELD (instance variable)
ElementType.METHOD
ElementType.PARAMETER
ElementType.CONSTRUCTOR
ElementType.LOCAL_VARIABLE
ElementType.ANNOTATION_TYPE (on another annotation)
ElementType.PACKAGE (remember package-info.java)

@Inherited – 控制是否影响子类

注解的内容(属性)仅支持String和enum类型,所有属性被定义成方法,还可以提供默认值

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@interface Todo {
public enum Priority {LOW, MEDIUM, HIGH}
public enum Status {STARTED, NOT_STARTED}
String author() default "Yash";
Priority priority() default Priority.LOW;
Status status() default Status.NOT_STARTED;
}

那么@Todo注解的使用方式是怎么呢,如下

@Todo(priority = Todo.Priority.MEDIUM, author = "Yashwant", status = Todo.Status.STARTED)
public void incompleteMethod1() {
//Some business logic is written
//But it’s not complete yet
}

如果Annotation里面仅有一个属性,可以命名为"value",如下

@interface Author{
String value();
}
@Author("Yashwant")
public void someMethod() {
}

How to use custom Annotations

1, 反射

反射会提供Class、Method和Field对象,这些对象都有共同的方法getAnnotation(),强转成自定义的Annotation后就可以使用在Annotation内部定义的属性,如下

Class businessLogicClass = BusinessLogic.class;
for(Method method : businessLogicClass.getMethods()) {
Todo todoAnnotation = (Todo)method.getAnnotation(Todo.class);
if(todoAnnotation != null) {
System.out.println(" Method Name : " + method.getName());
System.out.println(" Author : " + todoAnnotation.author());
System.out.println(" Priority : " + todoAnnotation.priority());
System.out.println(" Status : " + todoAnnotation.status());
}
}
## 2,Android里的@StringDef@IntDef代替Enum
[点这查看](http://tools.android.com/tech-docs/support-annotations)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 什么是注解(Annotation):Annotation(注解)就是Java提供了一种元程序中的元素关联任何信息和...
    九尾喵的薛定谔阅读 8,430评论 0 2
  • 本文章涉及代码已放到github上annotation-study 1.Annotation为何而来 What:A...
    zlcook阅读 29,402评论 15 116
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,390评论 19 139
  • 不同服务器上的不同数据库表结构,字段互相拷贝 1.不同服务器上,同一类数据库,postgresql数据库之间互相拷...
    邓凯献阅读 10,222评论 0 1
  • 注册这个账号想要用它来记录自己的一些想法,和向别人去探讨一些思想。 也就是在不久之前,自己慢慢意识到,之前的含蓄低...
    livereur阅读 1,085评论 0 0