欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Android自定义注解(一)

程序员文章站 2022-06-17 10:27:02
...

为什么要写这个,因为前段时间看了一下AOP相关的一些内容,然后也是太久没写注解,看得有点那啥不顺畅,所以想对注解做个总结。

一.JAVA自带的注解

(1)Override 覆盖
(2)Deprecated 标记过期方法
(3)SuppressWarnings 屏蔽警告

二.自定义注解

1.定义

我这里写个demo自定义一个注解

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface BindView {
    int value() default -1;
}

可以看出注解用@interface来标志。

2.元注解

可以看出在上边的定义中,上面还有两个注解,这些被称为元注解,什么是元注解,简单来说就是描述注解的注解
还有什么元数据的,元什么什么的,这个元其实不太好解释,我记得之前看过一个元数据的描述,英文是 data about data,这个元有这个about的那种感觉

元注解有4种
(1)@Retention 用来描述周期,Retention 有“保持时间”的意思,这个属性可以选三个值
SOURCE表示只在源码中有用,编译就没用了。
CLASS表示在编译中可用,运行就没用了。
RUNTIME表示运行时可用。
这三种的区别解释起来很麻烦,总之一搬我们都是使用RUNTIME

(2)@Target 用来描述作用域,Target有目标的意思,你这个注解要给哪个目标修饰,这个属性可以选以下的值
CONSTRUCTOR用于描述构造器
FIELD用于描述域
LOCAL_VARIABLE用于描述局部变量
METHOD用于描述方法
PACKAGE用于描述包
PARAMETER用于描述参数
这个就是说,你允许把注解写在什么地方。这个属性是可以多选的,比如@Target({ElementType.FIELD, ElementType.METHOD})

(3)@Inherited 描述是否可以为继承,默认是false

(4)@Documented 描述是否会保存到 Javadoc 文档中

一般我们只会用到前面两个,所以后面两个我就不解释了。

3.注解的值

可以给注解设置值,比如说我上面的代码,就在注解里写了个

int value() default -1;

表示在使用注解时需要传一个整形的值

@BindView(R.id.tv)
TextView textView;

我这传了个R.id.tv就是一个整形的值。只有一个值的时候,必须以** value()**来命名,然后调用时就直接传就行。我还是分情况来说吧。

(1)不需要传值的情况

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface BindView {

}

调用

@BindView
TextView textView;

(2)传一个值的情况
就是我上面写的代码

(3)传多个值的情况

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface BindView {
    int age;
    String name;
}

调用

@BindView(age = 18, name = "JackMa")
TextView textView;

最后,这个default 表示设置默认值的意思。

三.调用注解

上面我们创建出了注解并定义好,然后我们需要调用这个注解,那怎么调用呢,一般我们调用一个类或接口都是new 出来,但是这东西怎么看都觉得new不出来吧,所以我们需要用反射来获取到注解的对象,然后进行调用

我不想去讲反射的内容,不然就没完了。就说个思路,需要用到哪个方法可以去查api。
注解在Java中就是Annotation懂吧,所以在反射中和Annotation相关的方法都是和注解相关的方法。

比如说获取注解,我就可以调用.getAnnotation(BindView.class)
再比如说我判断注解存在不,可以调用isAnnotationPresent()
总之你只要记住注解是Annotation,之后你不管是在反射中还是在哪里找和注解相关的方法,都先往Annotation这个名词相关的地方找准没错。

如果你不懂反射这里说再多也没有。

四.案例

其实说了这么多,到底哪里需要使用到注解呢,如果不使用到,讲它有啥用,我个人是目前接触到AOP涉及注解比较频繁,其它时候,比如说ButterKnife,Dagger,Retrofit等等这些主流的框架都会用到注解,包括java后台,我接触过spring会涉及到IOC啊AOP啊这些思想用java来实现也是用到注解比较多,所以可以来写个小demo试试。

不如就仿照ButterKnife吧,我之前没看过ButterKnife的源码啊,我就按照我学会的注解的知识和ButterKnife的调用方法来仿写个ButterKnife

1.首先定义注解
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface BindView {
    int value() default -1;
}

定义注解,要求传个资源进来,默认是-1

2.按照ButterKnife的调用的样子来写调用的地方
public class MainActivity extends AppCompatActivity {

    @BindView(R.id.tv)
    TextView textView;
    @BindView(R.id.btn)
    Button btn;

    String yyy = "aasdasdas";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);

        textView.setText("wwwwwwwwwwwwww");
        btn.setText("btn");
    }
}

发现ButterKnife调用的地方使用这 ButterKnife.bind(this); 那我就推测findViewById就在这个ButterKnife类的bind方法中进行。

3.定义ButterKnife
public class ButterKnife {

    public static void bind(Activity activity){
        Class<?> cls = activity.getClass();
        Field[] fields = cls.getDeclaredFields();
        for (Field field : fields) {
            BindView bindView = field.getAnnotation(BindView.class);
            if (bindView != null) {
                int ids = bindView.value();
                try {
                    field.set(activity, activity.findViewById(ids));
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }
    }

}

我这里就用了反射来操作,调用bind方法后,先用cls.getDeclaredFields();获取所有的全局变量,然后再对每个变量用field.getAnnotation(BindView.class);来获取这个变量上面的注解,如果存在的话 int ids = bindView.value(); 来获取我们传入的资源,最后调用findViewById

field.set(activity, activity.findViewById(ids));

很简单吧,三个类这样就能实现了ButterKnife的效果


Android自定义注解(一)
image.png

注意判空一定要写,即使你全部的变量都用了注解,但是如果你DeBug看看,你会发现,变量中不仅仅只有你定义的,还存在其他的。

其实上面都是瞎写的,ButterKnife的原理根本不是这样,虽然我没看过源码,但我听别人说过ButterKnife的实现并不是这样的,我这的demo只是仿照了这个功能,具体ButterKnife的原理以后如果有时间我会单独写。

通过上面的讲解就能简单的实现注解,我也顺便复习加总结了一遍。