Annotation——注解,JDK1.5新增加的功能。它能够添加到 Java 源代码的语法元数据。类、方法、变量、参数、包都可以被注解,可用来将信息元数据与程序元素进行关联。目前很多开源库都使用到了注解,最熟悉的ButtonKnife中的@ViewInject(R.id.x)就可以替代findViewId,不懂这一块技术的同学第一眼看上去肯定会一脸懵逼,下面会手把手带大家写出ButtonKnife的注解使用。使用注解可以简化代码,提高开发效率。本文简单介绍下注解的使用,并对几个 Android 开源库的注解使用原理进行简析。
1、作用
2、分类
例如,注解@MethodInfo:
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Inherited
public @interface MethodInfo {
String author() default "[email protected]";
String date();
int version() default 1;
}
使用到了元Annotation:
注解的参数名为注解类的方法名,且:
public class App {
@MethodInfo(
author = “[email protected]”,
date = "2011/01/11",
version = 2)
public String getAppName() {
return "appname";
}
}
调用自定义MethodInfo 的示例,这里注解的作用实际是给方法添加相关信息: author、date、version 。
首先,先定义一个ViewInject注解。
public @interface ViewInject {
int value() default -1;
}
紧接着,为刚自定义注解添加元注解。
@Target({ElementType.FIELD, ElementType.PARAMETER, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewInject {
int value() default -1;
}
再定义一个注解LayoutInject
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface LayoutInject {
int value() default -1;
}
定义一个基础的Activity。
package cn.wsy.myretrofit.annotation;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import java.lang.reflect.Field;
public class InjectActivity extends AppCompatActivity {
private int mLayoutId = -1;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
displayInjectLayout();
displayInjectView();
}
/**
* 解析注解view id
*/
private void displayInjectView() {
if (mLayoutId <=0){return ;}
Class> clazz = this.getClass();
Field[] fields = clazz.getDeclaredFields();//获得声明的成员变量
for (Field field : fields) {
//判断是否有注解
try {
if (field.getAnnotations() != null) {
if (field.isAnnotationPresent(ViewInject.class)) {//如果属于这个注解
//为这个控件设置属性
field.setAccessible(true);//允许修改反射属性
ViewInject inject = field.getAnnotation(ViewInject.class);
field.set(this, this.findViewById(inject.value()));
}
}
} catch (Exception e) {
Log.e("wusy", "not found view id!");
}
}
}
/**
* 注解布局Layout id
*/
private void displayInjectLayout() {
Class> clazz = this.getClass();
if (clazz.getAnnotations() != null){
if (clazz.isAnnotationPresent(LayouyInject.class)){
LayouyInject inject = clazz.getAnnotation(LayouyInject.class);
mLayoutId = inject.value();
setContentView(mLayoutId);
}
}
}
}
首先,这里是根据映射实现设置控件的注解,java中使用反射的机制效率性能并不高。这里只是举例子实现注解。ButterKnife官方申明不是通过反射机制,因此效率会高点。
package cn.wsy.myretrofit;
import android.os.Bundle;
import android.widget.TextView;
import cn.wsy.myretrofit.annotation.InjectActivity;
import cn.wsy.myretrofit.annotation.LayouyInject;
import cn.wsy.myretrofit.annotation.ViewInject;
@LayoutInject(R.layout.activity_main)
public class MainActivity extends InjectActivity {
@ViewInject(R.id.textview)
private TextView textView;
@ViewInject(R.id.textview1)
private TextView textview1;
@ViewInject(R.id.textview2)
private TextView textview2;
@ViewInject(R.id.textview3)
private TextView textview3;
@ViewInject(R.id.textview4)
private TextView textview4;
@ViewInject(R.id.textview5)
private TextView textview5;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//设置属性
textView.setText("OK");
textview1.setText("OK1");
textview2.setText("OK2");
textview3.setText("OK3");
textview4.setText("OK4");
textview5.setText("OK5");
}
}
上面直接继承InjectActivity即可,文章上面也有说过:LayouyInject为什么作用域是TYPE,首先在加载view的时候,肯定是优先加载布局啊,**ButterKnife**也不例外。因此选择作用域在描述类,并且存在运行时。
(1) 运行时 Annotation 指 @Retention 为 RUNTIME 的 Annotation,可手动调用下面常用 API 解析
method.getAnnotation(AnnotationName.class);
method.getAnnotations();
method.isAnnotationPresent(AnnotationName.class);
其他 @Target 如 Field,Class 方法类似 。
(2) 解析示例如下:
public static void main(String[] args) {
try {
Class cls = Class.forName("cn.trinea.java.test.annotation.App");
for (Method method : cls.getMethods()) {
MethodInfo methodInfo = method.getAnnotation(
MethodInfo.class);
if (methodInfo != null) {
System.out.println("method name:" + method.getName());
System.out.println("method author:" + methodInfo.author());
System.out.println("method version:" + methodInfo.version());
System.out.println("method date:" + methodInfo.date());
}
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
以之前自定义的 MethodInfo 为例,利用 Target(这里是 Method)getAnnotation 函数得到 Annotation 信息,然后就可以调用 Annotation 的方法得到响应属性值 。
(1) 编译时 Annotation 指 @Retention 为 CLASS 的 Annotation,甴 apt(Annotation Processing Tool) 解析自动解析。
使用方法:
(2) 假设之前自定义的 MethodInfo 的 @Retention 为 CLASS,解析示例如下:
@SupportedAnnotationTypes({ "cn.trinea.java.test.annotation.MethodInfo" })
public class MethodInfoProcessor extends AbstractProcessor {
@Override
public boolean process(Set extends TypeElement> annotations, RoundEnvironment env) {
HashMap map = new HashMap();
for (TypeElement te : annotations) {
for (Element element : env.getElementsAnnotatedWith(te)) {
MethodInfo methodInfo = element.getAnnotation(MethodInfo.class);
map.put(element.getEnclosingElement().toString(), methodInfo.author());
}
}
return false;
}
}
SupportedAnnotationTypes 表示这个 Processor 要处理的 Annotation 名字。
process 函数中参数 annotations 表示待处理的 Annotations,参数 env 表示当前或是之前的运行环境
process 函数返回值表示这组 annotations 是否被这个 Processor 接受,如果接受后续子的 rocessor 不会再对这个 Annotations 进行处理
(1) 调用
@GET("/users/{username}")
User getUser(@Path("username") String username);
(2) 定义
@Documented
@Target(METHOD)
@Retention(RUNTIME)
@RestMethod("GET")
public @interface GET {
String value();
}
从定义可看出 Retrofit 的 Get Annotation 是运行时 Annotation,并且只能用于修饰 Method
(3) 原理
private void parseMethodAnnotations() {
for (Annotation methodAnnotation : method.getAnnotations()) {
Class extends Annotation> annotationType = methodAnnotation.annotationType();
RestMethod methodInfo = null;
for (Annotation innerAnnotation : annotationType.getAnnotations()) {
if (RestMethod.class == innerAnnotation.annotationType()) {
methodInfo = (RestMethod) innerAnnotation;
break;
}
}
……
}
}
RestMethodInfo.java 的 parseMethodAnnotations 方法如上,会检查每个方法的每个 Annotation, 看是否被 RestMethod 这个 Annotation 修饰的 Annotation 修饰,这个有点绕,就是是否被 GET、DELETE、POST、PUT、HEAD、PATCH 这些 Annotation 修饰,然后得到 Annotation 信息,在对接口进行动态代理时会掉用到这些 Annotation 信息从而完成调用。 因为 Retrofit 原理设计到动态代理,这里只介绍 Annotation。
(1) 调用
@InjectView(R.id.user)
EditText username;
(2) 定义
@Retention(CLASS)
@Target(FIELD)
public @interface InjectView {
int value();
}
可看出 Butter Knife 的 InjectView Annotation 是编译时 Annotation,并且只能用于修饰属性
(3) 原理
@Override
public boolean process(Set extends TypeElement> elements, RoundEnvironment env) {
Map targetClassMap = findAndParseTargets(env);
for (Map.Entry entry : targetClassMap.entrySet()) {
TypeElement typeElement = entry.getKey();
ViewInjector viewInjector = entry.getValue();
try {
JavaFileObject jfo = filer.createSourceFile(viewInjector.getFqcn(), typeElement);
Writer writer = jfo.openWriter();
writer.write(viewInjector.brewJava());
writer.flush();
writer.close();
} catch (IOException e) {
error(typeElement, "Unable to write injector for type %s: %s", typeElement, e.getMessage());
}
}
return true;
}
ButterKnifeProcessor.java 的 process 方法如上,编译时,在此方法中过滤 InjectView 这个 Annotation 到 targetClassMap 后,会根据 targetClassMap 中元素生成不同的 class 文件到最终的 APK 中,然后在运行时调用 ButterKnife.inject(x) 函数时会到之前编译时生成的类中去找。
(1) 调用
@Column(name = “Name")
public String name;
(2) 定义
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Column {
……
}
可看出 ActiveAndroid 的 Column Annotation 是运行时 Annotation,并且只能用于修饰属性
(3) 原理
Field idField = getIdField(type);
mColumnNames.put(idField, mIdName);
List fields = new LinkedList(ReflectionUtils.getDeclaredColumnFields(type));
Collections.reverse(fields);
for (Field field : fields) {
if (field.isAnnotationPresent(Column.class)) {
final Column columnAnnotation = field.getAnnotation(Column.class);
String columnName = columnAnnotation.name();
if (TextUtils.isEmpty(columnName)) {
columnName = field.getName();
}
mColumnNames.put(field, columnName);
}
}
TableInfo.java 的构造函数如上,运行时,得到所有行信息并存储起来用来构件表信息。
————————————————————————
最后一个问题,看看这段代码最后运行结果:
public class Person {
private int id;
private String name;
public Person(int id, String name) {
this.id = id;
this.name = name;
}
public boolean equals(Person person) {
return person.id == id;
}
public int hashCode() {
return id;
}
public static void main(String[] args) {
Set set = new HashSet();
for (int i = 0; i < 10; i++) {
set.add(new Person(i, "Jim"));
}
System.out.println(set.size());
}
}
答案:示例代码运行结果应该是 10 而不是 1,这个示例代码程序实际想说明的是标记型注解 Override 的作用,为 equals 方法加上 Override 注解就知道 equals 方法的重载是错误的,参数不对。