本文共 3901 字,大约阅读时间需要 13 分钟。
为什么要写这个,因为前段时间看了一下AOP相关的一些内容,然后也是太久没写注解,看得有点那啥不顺畅,所以想对注解做个总结。
(1)Override 覆盖
(2)Deprecated 标记过期方法 (3)SuppressWarnings 屏蔽警告我这里写个demo自定义一个注解
@Retention(RetentionPolicy.RUNTIME)@Target({ElementType.FIELD})public @interface BindView { int value() default -1;}
可以看出注解用@interface来标志。
可以看出在上边的定义中,上面还有两个注解,这些被称为元注解,什么是元注解,简单来说就是描述注解的注解
还有什么元数据的,元什么什么的,这个元其实不太好解释,我记得之前看过一个元数据的描述,英文是 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 文档中
一般我们只会用到前面两个,所以后面两个我就不解释了。
可以给注解设置值,比如说我上面的代码,就在注解里写了个
int value() default -1;
表示在使用注解时需要传一个整形的值
@BindView(R.id.tv)TextView textView;
我这传了个R.id.tv就是一个整形的值。只有一个值的时候,必须以** value()**来命名,然后调用时就直接传就行。我还是分情况来说吧。
(1)不需要传值的情况
@Retention(RetentionPolicy.RUNTIME)@Target({ElementType.FIELD})public @interface BindView {}
调用
@BindViewTextView 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
@Retention(RetentionPolicy.RUNTIME)@Target({ElementType.FIELD})public @interface BindView { int value() default -1;}
定义注解,要求传个资源进来,默认是-1
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方法中进行。
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的效果
注意判空一定要写,即使你全部的变量都用了注解,但是如果你DeBug看看,你会发现,变量中不仅仅只有你定义的,还存在其他的。
其实上面都是瞎写的,ButterKnife的原理根本不是这样,虽然我没看过源码,但我听别人说过ButterKnife的实现并不是这样的,我这的demo只是仿照了这个功能,具体ButterKnife的原理以后如果有时间我会单独写。
通过上面的讲解就能简单的实现注解,我也顺便复习加总结了一遍。
转载地址:http://cmykl.baihongyu.com/