Android注解(annotation)介绍及其应用

2023-04-10 11:55:11

我的新书《Android App开发入门与实战》已于2020年8月由人民邮电出版社出版,欢迎购买。点击进入详情

介绍

Annotation,又叫注解,是附加在代码上的元信息(用于标记属性)。
Annotation适用于IDE工具在编译、运行时对其解析和使用,起到配置的作用。
我们最常见的Annotation:

@Override
    public void onCreate() {
        super.onCreate();
    }

Override表明这个方法是一个重写的方法,需要IDE协助检查这个方法是否是父类的方法,如果不是则会报错提醒。如果不加Override,这个方法名不是父类的话,IDE也不会报错提醒。

从上面Override的例子我们可以看到,annotation更像是一个标签,它不仅仅可以附着在方法上面,还可以在包、类、字段、方法的参数、局部变量等上面进行标记。

自定义注解

还是用Override这个注解为例,我们看下源码:

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

这里要提到元注解的概念。什么是元注解?元注解就是注解上的注解,也就是注解上的标签。比如我们在自定义注解的时候,@Target、@Retention这些都是元注解。
那元注解有什么用的呢?我看不妨来看一下元注解的作用:
@Retention:注解生命周期范围。
@Target:注解对象的作用范围。
@Inherited:@Inherited标明子类是否可以继承该父类的注解。
@Documented:javadoc工具的文档化。

@Retention的值:
@Retention(RetentionPolicy.SOURCE) :注释仅存在源码中,class文件中不包含;
@Retention(RetentionPolicy.CLASS):默认策略,class文件中包含,运行时遗弃;
@Retention(RetentionPolicy.RUNTIME):class文件包含,运行时通过反射也可以获得;

@Target有很多值,就像上面提到的,注释不仅可以用在方法上,也可以用在包、类、字段等等其它上面。像@Override的@Target(ElementType.METHOD)标明注解的作用范围是方法。

注解的应用:反射

通过反射获取注解是一个很常用的方法,但是需要注意的是,如果通过反射获取Annotation,那么这个Annotation的@Retention策略需要是RUNTIME的。这样就导致了反射获取注解应用范围有限。
定义一个@ByString的注解:

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ByString {
    int value();
}

在源码中标记这个注解:

@ByString(R.string.annotation_reflection)
    private String text;

通过反射获取field:

Field[] fields = clazz.getDeclaredFields();
if (fields != null) {
    for (Field field : fields) {
	...
		}
}

找到这个注解:

ByString stringById = field.getAnnotation(ByString.class);
        if (stringById != null) {
            int stringId = stringById.value();
}

最后通过field的set方法进行赋值。

注解的应用:APT

APT(Annotation Processing Tool),它可以对源码进行扫描检测,找出其中的Annotation,并且自动生成相应的代码,并且将这些代码保存在额外的文件中(.class文件),最终这些.class文件一并参与apk的编译。
一个完整的APT包含:
1、AbstractProcessor:注解处理器,这是一个抽象类,一般需要自己实现一个。

public class TagBinderProcessor extends AbstractProcessor { ...}

2、处理器注册(AutoService):

@AutoService(Processor.class)
public class TagBinderProcessor extends AbstractProcessor { ...}

我们自定义了一个叫TagBinderProcessor的注解处理器,需要通过@AutoService注册一下。
3、代码自动生成(javaPoet)
根据注解,我们需要生成代码。而 JavaPoet 就是用来生成 Java 代码的一个 Java Library。
使用的时候需要引入:

implementation ‘com.squareup:javapoet:1.10.0’

我们看一段JavaPoet用法:

class AnnotatedClass {

    private ArrayList<BindTagField> mFields;

    AnnotatedClass() {
        mFields = new ArrayList<>();
    }

    void addField(BindTagField field) {
        mFields.add(field);
    }

    JavaFile generateFile() {
        //方法参数类型
        ParameterizedTypeName parameterizedTypeName = ParameterizedTypeName.get(
                ClassName.get(Map.class),
                ClassName.get(String.class),
                ClassName.get(TagInfo.class));
        ParameterSpec params = ParameterSpec.builder(parameterizedTypeName, "params").build();

        MethodSpec.Builder methodBuilder = MethodSpec.methodBuilder("load")
                .addModifiers(Modifier.PUBLIC)
                .addAnnotation(Override.class)
                .addParameter(params);

        for (BindTagField item : mFields) {
            String key = item.getTypeName().toString();
            TagInfo.Type type = item.getType();
            String[] value = item.getTag();
            String description = item.getDescription();
            // 添加方法内容
            methodBuilder.addStatement("params.put($S,$T.build($T.$L,$S,$S,$S))",
                    key,
                    ClassName.get(TagInfo.class),
                    ClassName.get(TagInfo.Type.class),
                    type,
                    key,
                    Arrays.toString(value),
                    description);
        }
        //生成类
        TypeSpec finderClass = TypeSpec.classBuilder("TagService")
                .addSuperinterface(ClassName.get("com.androidwind.annotation.core", "ILoad"))
                .addModifiers(Modifier.PUBLIC)
                .addMethod(methodBuilder.build())
                .build();

        return JavaFile.builder("com.androidwind.annotation", finderClass).build();
    }
}

Ctrl+B,项目Build一下,看见在如下目录下有一个TagService文件生成:
在这里插入图片描述
打开看一下里面的内容:

public class TagService implements ILoad {
  @Override
  public void load(Map<String, TagInfo> params) {
    params.put("la.xiong.androidquick.demo.features.architecture.mvc.MVCActivity",TagInfo.build(TagInfo.Type.ACTIVITY,"la.xiong.androidquick.demo.features.architecture.mvc.MVCActivity","[MVC]","Activity + MVC实例"));
    params.put("la.xiong.androidquick.demo.features.architecture.mvp.activity.MVPActivity",TagInfo.build(TagInfo.Type.ACTIVITY,"la.xiong.androidquick.demo.features.architecture.mvp.activity.MVPActivity","[MVP]","Activity + MVP实例"));
}
}

这个就是通过JavaPoet生成的代码内容。

其它

除了上面提到注解的应用,其实注解在很多场景下都能看到它的身影,比如AOP(面向切面编程),后续会有专门的文章来介绍这个。

  • 作者:元架构师
  • 原文链接:https://blog.csdn.net/ddnosh/article/details/100059679
    更新时间:2023-04-10 11:55:11