Java 编译时注解 —— 入门教程

简介: Java 编译时注解 —— 入门教程

为什么要写这一系列的博客呢?


因为在 Android 开发的过程中, 泛型,反射,注解这些知识进场会用到,几乎所有的框架至少都会用到上面的一两种知识,如 Gson 就用到泛型,反射,注解,Retrofit 也用到泛型,反射,注解 。学好这些知识对我们进阶非常重要,尤其是阅读开源框架源码或者自己开发开源框架。


java Type 详解

java 反射机制详解

注解使用入门(一)

Android 自定义编译时注解1 - 简单的例子

Android 编译时注解 —— 语法详解

带你读懂 ButterKnife 的源码


经过前面的介绍,相信大家对注解有了一定的了解了。


根据注解使用方法和用途,我们可以将Annotation分为三类:


  • JDK内置系统注解(如 @SuppressWarnings(“deprecation”),@override 等)
  • 元注解 如(@Documented ,@Retention() ,@Target(),@Documented )
  • 自定义注解 (自己实现的的注解)


元注解


元注解 解析说明


  • @Documented 是否会保存到 Javadoc 文档中


  • @Retention 保留时间,可选值, 默认为 CLASS


SOURCE(源码时),CLASS(编译时),RUNTIME(运行时)


  • @Target 可以用来修饰哪些程序元素,如 TYPE, METHOD, CONSTRUCTOR, FIELD, PARAMETER 等,未标注则表示可修饰所有
ANONOTATION_TYPE(注解类型声明),
PACKAGE(包)
TYPE (类,包括enum及接口,注解类型)
METHOD (方法)
CONSTRUCTOR (构造方法)
FIFLD (成员变量)
PARAMATER (参数)
LOCAL_VARIABLE (局部 变量)
  • @Inherited 是否可以被继承,默认为 false


需要注意的是注解是不可以继承的,@Inherited 的意思是 加入我们把注解应用到 A 类中,B 类继承 A ,那么 B 可以扫描到 A 的注解。


注解的继承”(依赖倒置?)


这里讲的继承并不是通过@Inherited修饰的注解。

这个“继承”是一个注解的使用技巧,使用上的感觉类似于依赖倒置,来自于ButterKnife源码。

先看代码。

@Target(METHOD)
@Retention(CLASS)
@ListenerClass(
    targetType = "android.view.View",
    setter = "setOnClickListener",
    type = "butterknife.internal.DebouncingOnClickListener",
    method = @ListenerMethod(
        name = "doClick",
        parameters = "android.view.View"
    )
)
public @interface OnClick {
      /** View IDs to which the method will be bound. */
      int[] value() default { View.NO_ID };
}


这是ButterKnife的OnClick 注解。特殊的地方在于@OnClick修饰了注解@ListenerClass,并且设置了一些只属于@OnClick的属性。


那这样的作用是什么呢?


凡是修饰了@OnClick的地方,也就自动修饰了@ListenerClass。类似于@OnClick是@ListenerClass的子类。而ButterKnife有很多的监听注解@OnItemClick、@OnLongClick等等。


这样在做代码生成时,不需要再单独考虑每一个监听注解,只需要处理@ListenerClass就OK。如 @interface  OnItemClick 等。

@Target(METHOD)
@Retention(CLASS)
@ListenerClass(
    targetType = "android.widget.AdapterView<?>",
    setter = "setOnItemClickListener",
    type = "android.widget.AdapterView.OnItemClickListener",
    method = @ListenerMethod(
        name = "onItemClick",
        parameters = {
            "android.widget.AdapterView<?>",
            "android.view.View",
            "int",
            "long"
        }
    )
)
public @interface OnItemClick {
  /** View IDs to which the method will be bound. */
  @IdRes int[] value() default { View.NO_ID };
}

自定义注解


一个简单的自定义注解例子

@Documented()
// 表示是基于编译时注解的
@Retention(RetentionPolicy.CLASS)
// 表示可以作用于成员变量,类、接口
@Target({ElementType.FIELD, ElementType.TYPE}) 
public @interface Seriable {
}


指定默认值

@Documented()
// 表示是基于编译时注解的
@Retention(RetentionPolicy.CLASS)
// 表示可以作用于成员变量,类、接口
@Target({ElementType.FIELD, ElementType.TYPE}) 
public @interface Seriable {
     int id();
     String name() default "test";
}
//使用
@Seriable(id = 1) //name有默认值可以不写
class Test{
}

关于怎样自定义一个注解,可以参看这一篇博客,Android 自定义编译时注解1 - 简单的例子


处理器类Processor编写


自定义注解后,需要编写Processor类处理注解。Processor 继承自 AbstractProcessor 的类。我们主要需要处理以下连个方法。


  • public Set getSupportedAnnotationTypes()
  • public abstract boolean process(Set<? extends TypeElement> annotations,
  • RoundEnvironment roundEnv);


getSupportedAnnotationTypes 方法


重写 getSupportedAnnotationTypes 方法:告知Processor哪些注解需要处理。返回一个Set集合,集合内容为 自定义注解的包名+类名

建议项目中这样编写:

@Override
public Set<String> getSupportedAnnotationTypes() {
    Set<String> types = new LinkedHashSet<>();
    //需要全类名
    types.add(Seriable.class.getCanonicalName()); 
    types.add(Println.class.getCanonicalName());
    return types;
}

另外如果注解数量很少的话,可以通过另一种方式实现:

//在只有一到两个注解需要处理时,可以这样编写:
@SupportedAnnotationTypes({"com.example.Seriable"})
@SupportedSourceVersion(SourceVersion.RELEASE_7)
public class JsonProcessor extends AbstractProcessor {
}


process 方法


process 方法,这个方法是所有方法中最关键的一个方法,他用来处理注解的相关信息,比如提取注解的信息,存进 map 集合或者生成代码等。

@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
    return false;
}
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
    //   第一步,根据我们自定义的注解拿到 elememts set 集合
    Set<? extends Element> elememts = roundEnv.getElementsAnnotatedWith(Seriable.class);
    TypeElement typeElement;
    VariableElement variableElement;
    //  第二步: 根据 element 的类型做相应的处理,并存进 map 集合
    for (Element element : elememts) {
        ElementKind kind = element.getKind();
        // 判断该元素是否为类
        if (kind == ElementKind.CLASS) {
            typeElement = (TypeElement) element;
            // 判断该元素是否为成员变量
        } else if (kind == ElementKind.FIELD) {
            variableElement = (VariableElement) element;
        }
    }
    return true;
}


Element


元素,虽有通过注解取得的元素都以 Element 等待处理,它的具体类型与我们通过 @Target 来标记的具有一定的联系。


官方的解释是这样的:


Represents a program element such as a package, class, or method.Each element represents a static, language-level construct

(and not, for example, a runtime construct of the virtual machine)


表示程序元素如包、类或者方法。每个元素代表一个静态语言级构造(例如,而不是运行时构建的虚拟机)

例如:

    // 第一步,根据我们自定义的注解拿到 elememts set 集合
    Set<? extends Element> elememts = roundEnv.getElementsAnnotatedWith(Seriable.class);
    TypeElement typeElement;
    VariableElement variableElement;
    //  第二步: 根据 element 的类型做相应的处理,并存进 map 集合
    for (Element element : elememts) {
        ElementKind kind = element.getKind();
        // 判断该元素是否为类
        if (kind == ElementKind.CLASS) {
            typeElement = (TypeElement) element;
            // 判断该元素是否为成员变量
        } else if (kind == ElementKind.FIELD) {
            variableElement = (VariableElement) element;
        }
    }


Element 的子类


Element 的子类有:


  • ExecutableElement


表示某个类或接口的方法、构造方法或初始化程序(静态或实例),包括注释类型元素。


对应@Target(ElementType.METHOD) @Target(ElementType.CONSTRUCTOR)


  • PackageElement;


表示一个包程序元素。提供对有关包极其成员的信息访问。


对应@Target(ElementType.PACKAGE)


  • TypeElement;


表示一个类或接口程序元素。提供对有关类型极其成员的信息访问。


对应@Target(ElementType.TYPE)


注意:枚举类型是一种类,而注解类型是一种接口。

  • TypeParameterElement;


表示一般类、接口、方法或构造方法元素的类型参数。


对应@Target(ElementType.PARAMETER)


  • VariableElement;


表示一个字段、enum常量、方法或构造方法参数、局部变量或异常参数。


对应 @Target(ElementType.LOCAL_VARIABLE)


判断 Element 具体属于哪一种子类


我们可以通过 element.getKind(); 来得到 Element 到底是哪一种类别,该方法返回 ElementKind 类型,我们在根据 ElementKind 即可得出 Element 到底是哪种类别。


ElementKind kind = element.getKind();
// 判断该元素是否为类
if (kind == ElementKind.CLASS) {
    typeElement = (TypeElement) element;
} else if (kind == ElementKind.FIELD) {
   variableElement = (VariableElement) element;
}


看到这里,想起前面的一篇博客  java Type 详解 ,我们需要注意 Type 与 Element 的区别。


Type 是用来处理泛型的,Element 是用来处理注解的。


题外话


注解语法暂时介绍到这里,以后想到新的,会逐步更新到这篇博客,下一篇,将分析 ButterKnife 源码。


相关文章
|
20天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
58 7
|
2月前
|
XML Java 编译器
Java学习十六—掌握注解:让编程更简单
Java 注解(Annotation)是一种特殊的语法结构,可以在代码中嵌入元数据。它们不直接影响代码的运行,但可以通过工具和框架提供额外的信息,帮助在编译、部署或运行时进行处理。
100 43
Java学习十六—掌握注解:让编程更简单
|
25天前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
66 5
|
1月前
|
Java 开发者 Spring
[Java]自定义注解
本文介绍了Java中的四个元注解(@Target、@Retention、@Documented、@Inherited)及其使用方法,并详细讲解了自定义注解的定义和使用细节。文章还提到了Spring框架中的@AliasFor注解,通过示例帮助读者更好地理解和应用这些注解。文中强调了注解的生命周期、继承性和文档化特性,适合初学者和进阶开发者参考。
62 14
|
1月前
|
前端开发 Java
[Java]讲解@CallerSensitive注解
本文介绍了 `@CallerSensitive` 注解及其作用,通过 `Reflection.getCallerClass()` 方法返回调用方的 Class 对象。文章还详细解释了如何通过配置 VM Options 使自定义类被启动类加载器加载,以识别该注解。涉及的 VM Options 包括 `-Xbootclasspath`、`-Xbootclasspath/a` 和 `-Xbootclasspath/p`。最后,推荐了几篇关于 ClassLoader 的详细文章,供读者进一步学习。
37 12
|
2月前
|
分布式计算 大数据 Java
大数据-86 Spark 集群 WordCount 用 Scala & Java 调用Spark 编译并打包上传运行 梦开始的地方
大数据-86 Spark 集群 WordCount 用 Scala & Java 调用Spark 编译并打包上传运行 梦开始的地方
42 1
大数据-86 Spark 集群 WordCount 用 Scala & Java 调用Spark 编译并打包上传运行 梦开始的地方
|
2月前
|
IDE Java 编译器
Java:如何确定编译和运行时类路径是否一致
类路径(Classpath)是JVM用于查找类文件的路径列表,对编译和运行Java程序至关重要。编译时通过`javac -classpath`指定,运行时通过`java -classpath`指定。IDE如Eclipse和IntelliJ IDEA也提供界面管理类路径。确保编译和运行时类路径一致,特别是外部库和项目内部类的路径设置。
196 5
|
2月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
59 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
1月前
|
Java 编译器
Java进阶之标准注解
Java进阶之标准注解
34 0
|
2月前
|
JSON Java 数据库
java 常用注解大全、注解笔记
关于Java常用注解的大全和笔记,涵盖了实体类、JSON处理、HTTP请求映射等多个方面的注解使用。
46 0
java 常用注解大全、注解笔记