Android,几分钟教你怎么应用自定义注解

来源:互联网 发布:专升本软件工程c语言 编辑:程序博客网 时间:2024/04/29 11:51

    相信各位Android程序猿都了解过 ButterKnife 这个高效的注解,对于 InjectView 高效的替代findViewId更是熟之又熟。以下代码:

@InjectView(R.id.textview)private TextView textView;

    好了,今天目的不是为了介绍 ButterKnife 这个框架哈。这次写的文章主要是为了介绍注解基本概念,同时用案例实现注解代替findViewId、setContentView。

一、注解作用

在Java开发中,注解一般有一下功能:
(1) 标识
    在jdk中,类似我们比较常见的注解有Override,Deprecated,SuppressWarnings,这些作用只是作为标识,删除对程序没影响。他们的作用分别为:
Override 表示这个方法重写了父类的方法
Deprecated 表示jdk中不建议使用这个方法或者属性
SuppressWarnings 表示屏蔽了某些警告

(2)运行时处理
    这个编译器默认的做法,编译器会通过class文件,逐个逐个的遍历class的属性和方法,即运行时处理。

(3)编译时处理
    在运行之前,有学过C的程序猿就会知道,编译器在运行之前,编译的时候会将include进来的*.h文件进行引入。同理例如你在代码中引入注解,编译器会在编译的时候,在注解的属性引入在进行编译。

二、注解基础

首先

这里先看下定义的一个注解

public @interface ViewInject {    int value() default  -1;}

    这上面记得要添加一个@喔,不然就变成了定义了一个接口。在这里,value()不是代表一个方法,而是代表一个属性,其中value是属于整形的属性,默认值为1。

其次

这里先看下定义的一个注解的时候,添加元注解

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

上面的代码举例了在定义注解的时候,常见的元注解Target和Retention,这里说明下:
(一)Target ->注解工表明应用在什么地方
1.CONSTRUCTOR:用于描述构造器
2.FIELD:用于描述域
3.LOCAL_VARIABLE:用于描述局部变量
4.METHOD:用于描述方法
5.PACKAGE:用于描述包
6.PARAMETER:用于描述参数
7.TYPE:用于描述类、接口(包括注解类型) 或enum声明
(二)Retention ->注解的功能差不多说明的就是你的注解的生命周期
1.SOURCE:在源文件中有效(即源文件保留)
2.CLASS:在class文件中有效(即class保留)
3.RUNTIME:在运行时有效(即运行时保留)

最后

综合上面的说明,以下代码说明:

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

    这个注解作用域在类中(待会看实例就知道为什么),并且只要在运行时就有效,定义了一个默认属性value。

三、实战注解

    在第一段的时候,我们说过ButterKnife注解可以直接秒杀findViewId,这一瞬间代码简洁了很多。现在我们动手不用框架,自己写一个注解实现这个功能,重在实践嘛~现在贴上代码,注释都写的比较清楚哈。

(一)定义一个基础的Activity

用于初始化解析注解,分别设置布局与初始化view,代码如下:

package cn.wsy.myretrofit.annotation;import android.os.Bundle;import android.support.v7.app.AppCompatActivity;import android.util.Log;import java.lang.reflect.Field;/** * Created by wsy on 2016/8/18. */public class InjectActivity extends AppCompatActivity {    private int mLayoutId = -1;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        displayInjectLayout();        displayInjectView();    }    /**     * 解析注解view id     */    private void displayInjectView() {        if (mLayoutId <=0){return ;}        Class<?> clazz = this.getClass();        Field[] fields = clazz.getDeclaredFields();//获得声明的成员变量        for (Field field : fields) {            //判断是否有注解            try {                if (field.getAnnotations() != null) {                    if (field.isAnnotationPresent(ViewInject.class)) {//如果属于这个注解                        //为这个控件设置属性                        field.setAccessible(true);//允许修改反射属性                        ViewInject inject = field.getAnnotation(ViewInject.class);                        field.set(this, this.findViewById(inject.value()));                    }                }            } catch (Exception e) {//                throw new InterruptedException("not found view id!");                Log.e("wusy", "not found view id!");            }        }    }    /**     * 注解布局Layout id     */    private void displayInjectLayout() {        Class<?> clazz = this.getClass();        if (clazz.getAnnotations() != null){            if (clazz.isAnnotationPresent(LayouyInject.class)){                LayouyInject inject = clazz.getAnnotation(LayouyInject.class);                mLayoutId = inject.value();                setContentView(mLayoutId);            }        }    }}

    首先,这里是根据映射实现设置控件的注解。作为程序猿的各位应该知道,java中使用反射的机制,效率性能并不高。所以真正开发的时候,需要根据需求去思考怎么样的实现方式哈,这里笔者只是举例子实现注解。ButterKnife官方申明不是通过反射机制,因此效率会高点。

(二)应用注解
package cn.wsy.myretrofit;import android.os.Bundle;import android.widget.TextView;import cn.wsy.myretrofit.annotation.InjectActivity;import cn.wsy.myretrofit.annotation.LayouyInject;import cn.wsy.myretrofit.annotation.ViewInject;@LayouyInject(R.layout.activity_main)public class MainActivity extends InjectActivity {    @ViewInject(R.id.textview)    private TextView textView;    @ViewInject(R.id.textview1)    private TextView textview1;    @ViewInject(R.id.textview2)    private TextView textview2;    @ViewInject(R.id.textview3)    private TextView textview3;    @ViewInject(R.id.textview4)    private TextView textview4;    @ViewInject(R.id.textview5)    private TextView textview5;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);       //设置属性        textView.setText("OK");        textview1.setText("OK1");        textview2.setText("OK2");        textview3.setText("OK3");        textview4.setText("OK4");        textview5.setText("OK5");    }}

    上面直接继承InjectActivity即可,文章上面也有说过:LayouyInject为什么作用域是TYPE,首先在加载view的时候,肯定是优先加载布局啊,ButterKnife也不例外。因此选择作用域在描述类,并且存在运行时。

三、总结

    今天大概归纳了日常注解常用的元注解,并且用反射机制的方式,实现了控件与布局的注解,希望可以帮到读者理解,谢谢!

傻小孩b mark
共勉,写给在成长路上奋斗的你

喜欢就为我点下喜欢吧:-D,感谢各位读者阅读。

0 0
原创粉丝点击