Android 编译时注解 —— 语法详解(1)

  • @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 getSupportedAnnotationTypes() {

Set 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;
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

https://i-blog.csdnimg.cn/blog_migrate/2daeb63ff863f82b6defe4318939257c.jpeg" />

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家

[外链图片转存中…(img-JhAhv5IB-1712526400757)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 26
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值