Java注解实现OnClickListener

在学习Android开发的过程中用到了许多的开源框架,很多框架都大量的用到了注解这个概念,虽然会用别人的框架,但是总觉得自己眼前有一片雾霾,所以决定去认识注解背后的原理,拨开这层迷雾。

刚开始开发Android的过程中经常写出下面的代码:

view.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        //do something...      
    }
});

直到后来用了ButterKnife变成了这样:

@OnClick(R.id.view)
public void viewClick(){
    //do something...     
}

遇到了这样的神器,心中激动不已,这是怎样神奇的技术啊,简化了劳资很多代码,现在我就要探寻这神秘背后的真实面目:注解

Java注解是在Java 1.5中出现的
概念:
Java 提供了一种原程序中的元素关联任何信息和任何元数据的途径和方法。(反正我是没看懂)
直观一点就是那些插入到源代码中使用其他工具可以对其进行处理的标签。(如:@Override)

Java中的常见注解
  • JDK自带注解
    1.@Override (被该标签修饰的方法是重写父类的方法)
    2.@Deprecated (表示该方法已经过时,不建议使用)
    3.@Suppvisewarnings (忽略一些特定的警告)

  • ButterKnife(Android常用的view注解框架)的注解
    1.@Bind (获得一个view的实例)
    2.@OnClick (添加点击事件)
    3.@OnItemSelected (添加Listview的点击事件)

注解不止这一些,还有很多,我只列举了常见的一些,大家看起来应该非常的熟悉

注解的分类(按运行机制分类)
  • 源码注解 (注解只在源代码中存在,编译成.class文件就不存在了)
  • 编译时注解 (注解在源码和.class文件中都存在)
  • 运行时注解 (在运行阶段还会起作用,甚至会影响运行逻辑的注解)
自定义注解
  • 注解语法
@Target()      //用来标注该注解使用的范围如类,方法等
@Retention()   //指定一条注解应该保留多长时间,源码级别,编译级别,运行时级别
modifiers @interface AnnotationName{
    type elementName1();//不带默认值
    type elementName2() default value;//带默认值
  }
  • 注解的注意事项
    1.一个注解接口的方法不能有任何参数和任何的throws语句,并且他们也不能是泛型的

2.注解的元素类型必须为以下几种

  • 基本数据类型 (int,short,long,byte,char,double,float,boolean)
  • String
  • Class (具有一个可选的类型参数,例如Class<? extends MyClass>)
  • enum类型
  • 注解类型
  • 由前面类型组成的数组 (由数组组成的数组不是合法的元素类型)

3.元注解

  • @Target({ElementType.TYPE,ElementType.METHOD})
  • ANNOTATION_TYPE 适用于注解类型声明
  • PACKAGE 适用于包
  • TYPE 适用于类以及接口(包括枚举及注解类型)
  • CONSTRUCTOR 适用于构造器
  • FIELD 成员属性(域)
  • PARAMETER 方法或构造器的参数
  • LOCAL_VARIABLE 局部变量
  • @Retention(RetentionPolicy.RUNTIME)
  • SOURCE 源码级别的注解,编译之后就不存在了
  • CLASS 编译时注解,但是需要载入虚拟机
  • RUNTIME 运行时注解,可通过反射API改变运行状态
  • @Inherited 表示该注解可被子类继承(一定是类,枚举和接口无效)
注解实现OnClickListener(Android)

上面介绍了一些基本概念,接下来就开始动手来实现这一监听效果吧!

  • 首先,定义一个注解接口
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface OnClick {
    int value() default -1;//只有一个元素用于获取View的id,当只有一个元素时,元素名必须为value
}
  • 然后,编写注解处理器(如果不理解反射请看上一篇文章Java反射机制
public class PanViewInjector {
    public static void process(final Object o) {
        Class c = o.getClass();
        Method[] methods = c.getDeclaredMethods();
        for (final Method m : methods) {
            OnClick click = m.getAnnotation(OnClick.class);//通过反射api获取方法上面的注解
            if (click != null) {
                if (o instanceof Activity) {
                    if (click.value() == -1) return;
                    View view = ((Activity) o).findViewById(click.value());//通过注解的值获取View控件
                    if (view == null) return;
                    view.setOnClickListener(new View.OnClickListener() {
                        @Override
                        public void onClick(View v) {
                            try {
                                m.invoke(o);//通过反射来调用被注解修饰的方法
                            } catch (Exception e) {
                                e.printStackTrace();
                            }
                        }
                    });
                }
            }
        }

    }
}
  • 最后在Activity里面使用注解,并在Oncreate里面调用PanViewInjector.process()
public class MainActivity extends AppCompatActivity {

    @OnClick(R.id.centerButton)//使用注解
    public void toCenter(){
        Toast.makeText(this,"注解起作用啦",Toast.LENGTH_SHORT).show();
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        PanViewInjector.process(this);//调用注解处理器
    }
}

到了这里,这个注解写的就差不多了,显然这几行代码跟ButterKnife是不能相提并论的,还有许多细节要处理,这是自定义注解最简单的用法,了解了注解的基本原理,我们才可以更加自信的去使用基于注解的一些开源框架。由于笔者水平有限,如有错误之处,请不吝赐教。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,026评论 19 139
  • 本文章涉及代码已放到github上annotation-study 1.Annotation为何而来 What:A...
    zlcook阅读 29,325评论 15 116
  • 什么是注解(Annotation):Annotation(注解)就是Java提供了一种元程序中的元素关联任何信息和...
    九尾喵的薛定谔阅读 3,243评论 0 2
  • 在经过一次没有准备的面试后,发现自己虽然写了两年的android代码,基础知识却忘的差不多了。这是程序员的大忌,没...
    猿来如痴阅读 2,892评论 3 10
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,785评论 18 399