Android 使用注解和反射自制简单版的butternife

一.Java反射机制。

1.反射机制的定义。

Java反射机制是指在运行状态中,
对于任意一个类,都能知道这个类的所有属性和方法;
对于任何一个对象,都能够调用它的任何一个方法和属性;
这样动态获取新的以及动态调用对象方法的功能就叫做反射。

2.反射机制的作用。

  • 在运行时判断任意一个对象所属的类;

  • 在运行时构造任意一个类的对象;

  • 在运行时判断任意一个类所具有的成员变量和方法;

  • 在运行时调用任意一个对象的方法;

3.获取字节码。

1. 获取字节码

方式 解析
Class clazz1 = Class.forName("全限定类名"); 通过Class类中的静态方法forName,直接获取到一个类的字节码文件对象,此时该类还是源文件阶段,并没有变为字节码文件。
Class clazz2 = Person.class; 当类被加载成.class文件时,此时Person类变成了.class,在获取该字节码文件对象,也就是获取自己, 该类处于字节码阶段。
Class clazz3 = p.getClass(); 通过类的实例获取该类的字节码文件对象,该类处于创建对象阶段

2. 实例化字节码
(1.) 无参构造。

 Class<?> aClass = Class.forName("com.demo.myapplication.Person");
  Person person = (Person) aClass.newInstance();//无参构造

(2.) 有参构造。

   Class<?> aClass = Class.forName("com.demo.myapplication.Person");
   Constructor<?> constructor = aClass.getConstructor(int.class, String.class);
   Person xiaoming = (Person) constructor.newInstance(10, "小明");

4.反射的实现。

例如有一个类:

package com.yousheng.demo;

public class Person {
    //私有属性
    private int age = 19;
    //公共属性
    public String name = "小明";

    //私有方法
    public String getName() {
        return name;
    }

    //私有方法
    private String getNameAndAge(String name, int age) {
        this.name = name;
        this.age = age;
        return "姓名为:" + name + "年龄为" + age;
    }
}

测试调用私有属性和私有方法。

  try {
            Class<?> aClass = Class.forName("com.yousheng.myapplication.Person");
            Person person = (Person) aClass.newInstance();
            //访问公共属性
            System.out.println("访问公共属性-->"+person.name);

            //访问私有属性
            Field ageField = aClass.getDeclaredField("age");
            //允许访问私有字段
            ageField.setAccessible(true);
            //获得私有字段值
            int age= (int) (ageField).get(person);
            System.out.println("访问私有属性-->"+age);

            //访问私有方法.
            //name为方法名
            //String.class 是第一个参数的类型
            //int.class 是第二个参数的类型
            Method getNameAndAge = aClass.getDeclaredMethod("getNameAndAge",String.class,int.class);
            //允许公共访问
            getNameAndAge.setAccessible(true);
            //调用方法
            String result = (String) getNameAndAge.invoke(person, "比尔盖茨", 52);
            System.out.println("访问私有方法-->"+result);

        } catch (Exception e) {
            e.printStackTrace();
        }

结果为:

    访问公共属性-->小明
    访问私有属性-->19
    访问私有方法-->姓名为:比尔盖茨年龄为52

二.java注解。

1.系统自带的注解。

例如:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}

2.元注解。

给自定义注解使用的注解就是元注解。
java中元注解有四个: @Retention @Target @Document @Inherited
1. @Target
表示该注解可以用于什么地方

  • ElementType.TYPE:作用于接口、类、枚举、注解,可以给一个类型进行注解,比如类、接口、枚举;
  • ElementType.FIELD:作用于成员变量(字段、枚举的常量),可以给属性进行注解;
  • ElementType.METHOD:作用于方法,可以给方法进行注解;
  • ElementType.PARAMETER:作用于方法的参数,可以给一个方法内的参数进行注解;
  • ElementType.CONSTRUCTOR:作用于构造函数,可以给构造方法进行注解;
  • ElementType.LOCAL_VARIABLE:作用于局部变量,可以给局部变量进行注解;
  • ElementType.ANNOTATION_TYPE:作用于Annotation,可以给一个注解进行注解。
  • ElementType.PACKAGE:作用于包名,可以给一个包进行注解;
  • ElementType.TYPE_PARAMETER:java8新增,但无法访问到;
  • ElementType.TYPE_USE:java8新增,但无法访问到;

2. @Retention

表示需要在什么级别保存该注解信息。可选的RetentionPolicy参数包括:

  • SOURCE:只在*.java源文件的时候有效,编译成class就没用了;

  • CLASS:只在.java或者.class中的文件有效,但是在运行时无效;

  • RUNTIME:VM将在运行期间保留注解,因此可以通过反射机制读取注解的信息。包含以上两种,并且运行时也会有效果,一般我们都会选用该参数。

3. @Document

将注解包含在Javadoc中

4. @Inherited

允许子类继承父类中的注解

3. 注解的属性

注解的属性也叫做成员变量。注解只有成员变量,没有方法。
注解的成员变量在注解的定义中以“无形参的方法”形式来声明,其方法名定义了该成员变量的名字,其返回值定义了该成员变量的类型。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Test {
    int id();
    String name();
}

上面代码定义了 Test 这个注解中拥有 id 和 name两个属性。在使用的时候,我们应该给它们进行赋值。

@TestAnnotation(id=3,name="hello annotation")
public class Test {
}

赋值的方式是在注解的括号内以 value=”” 形式,多个属性之前用 ,隔开。
在注解中一般会有一些元素以表示某些值。注解的元素看起来就像接口的方法,唯一的区别在于可以为其制定默认值。没有元素的注解称为标记注解。
注解中属性可以有默认值,默认值需要用 default 关键值指定。比如:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Test {
    public int id() default -1;
    public String name() default "Hello";
}

有默认值的话就可以不用进行赋值了。

@Test ()
public class Test {}

元素不能有不确定的值,即要么有默认值,要么在使用注解的时候提供元素的值。而且元素不能使用null作为默认值

另外,还有一种情况。如果一个注解内仅仅只有一个名字为 value 的属性时,应用这个注解时可以直接接属性值填写到括号内。

public @interface Test {
    String value();
}

上面代码中,Test 这个注解只有 value 这个属性。所以可以这样应用。

@Test ("hello")
int a;

这和下面的效果是一样的

@Test (value="hello")
int a;

最后,还需要注意的一种情况是一个注解没有任何属性。比如

public @interface Test {}

那么在应用这个注解的时候,括号都可以省略。

4、解析提取注解参数

Java通过反射机制获取类、方法、属性上的注解,因此java.lang.reflect提供AnnotationElement支持注解,主要方法如下:

方法 说明
boolean is AnnotationPresent(Class<?extends Annotation> annotationClass) 判断该元素是否被annotationClass注解修饰
<T extends Annotation> T getAnnotation(Class<T> annotationClass) 获取 该元素上annotationClass类型的注解,如果没有返回null
Annotation[] getAnnotations() 返回该元素上所有的注解
<T extends Annotation> T[] getAnnotationsByType(Class<T> annotationClass) 返回该元素上指定类型所有的注解
Annotation[] getDeclaredAnnotations() 返回直接修饰该元素的所有注解
<T extends Annotation> T[] getDeclaredAnnotationsByType(Class<T> annotationClass) 返回直接修饰该元素的所有注解

5.结合注解和反射写一个简单的版本的butternife。

运行时注解是通过反射来实现的,这种方式的效率会受到一定的影响,因此现在大多数的开源注解框架都是采用编译时注解的方式实现的,这种方式是在编译的时候生成所需的代码,不会影响运行的效率

方式一(运行时(Runtime)通过反射机制运行处理的注解):
  1. 先自定义两个注解。
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)//运行时注解
public @interface FindViewById {
    //使用value命名,则使用的时候可以忽略,否则使用时就得把参数名加上
    int value();
}
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface SetOnClickListener {
    int id();
    String methodName();
}
  1. 创建注解解析器。
public class AnnotationParse {
    public static void parse(final Activity activity) {
        try {
            //反射获取Class对象
            Class<? extends Activity> aClass = activity.getClass();
            //获取所有的字段
            Field[] declaredFields = aClass.getDeclaredFields();
            //遍历字段
            for (Field declaredField : declaredFields) {
                //获取字段的所有的注解信息
                Annotation[] annotations = declaredField.getAnnotations();
                //遍历注解信息
                for (Annotation annotation : annotations) {
                    if (annotation instanceof FindViewById) {
                        //注解类型为 FindViewById
                        FindViewById findViewByIdAnnotation = declaredField.getAnnotation(FindViewById.class);
                        //设置可修改该字段
                        declaredField.setAccessible(true);
                        //获取该注解value所对的值
                        int viewID = findViewByIdAnnotation.value();

                        View view = activity.findViewById(viewID);

                        if (view != null)
                            declaredField.set(activity, view);
                        else
                            throw new Exception("不能找到该View" + declaredField.getName() + ".");

                    } else if (annotation instanceof SetOnClickListener) {
                        //注解类型为 SetOnClickListener
                        SetOnClickListener clickAnnotation = declaredField.getAnnotation(SetOnClickListener.class);
                        //设置可修改该字段
                        declaredField.setAccessible(true);
                        //获取该注解id所对的值
                        int viewId = clickAnnotation.id();
                        //获取该注解的methodName所对应的值
                        String methodName = clickAnnotation.methodName();
                        //获取view
                        View view = (View) declaredField.get(activity);
                        if (view == null) {
                            // 如果对象为空,则重新查找对象
                            view = activity.findViewById(viewId);
                            if (view != null)
                                declaredField.set(activity, view);
                            else
                                throw new Exception("");
                        }
                        //获取目标对象的所有方法集
                        Method[] declaredMethods = aClass.getDeclaredMethods();
                        //遍历方法集
                        for (final Method declaredMethod : declaredMethods) {
                            //判断方法名是否和注解的methodName相同
                            if (declaredMethod.getName().equals(methodName)) {
                                //设置点击事件
                                view.setOnClickListener(new View.OnClickListener() {
                                    @Override
                                    public void onClick(View v) {
                                        try {
                                            //调用方法
                                            declaredMethod.invoke(activity);
                                        } catch (IllegalAccessException e) {
                                            e.printStackTrace();
                                        } catch (InvocationTargetException e) {
                                            e.printStackTrace();
                                        }
                                    }
                                });
                                break;
                            }
                        }
                    }
                }
            }
        } catch (Exception e) {
            System.out.println(e.getMessage());
        }
    }
}
  1. 调用
    mian_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/txt_main"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textColor="@color/colorPrimary"/>

    <Button
        android:id="@+id/btn_main"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="click me"/>

</LinearLayout>

MainActivity.java

 @FindViewById(R.id.txt_main)
    private TextView mTxtMain;
    @SetOnClickListener(id = R.id.btn_main, methodName = "onClick")
    private Button mBtnMain;

    /**
     * 点击事件
     */
    public void onClick() {
        Toast.makeText(this, "Hello world", Toast.LENGTH_SHORT).show();
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        AnnotationParse.parse(this);
        mTxtMain.setText("测试代码");
}
方式二(运行时(Runtime)通过反射机制运行处理的注解):

部分内容节选自:
java注解-最通俗易懂的讲解

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容

  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 8,983评论 0 13
  • 如果需要原文档(因文体限制,部分表格无法呈现)请联系QQ1769090563 本文由中医仲景协会整理收集 《内经选...
    陶墨阅读 34,307评论 0 33
  • 选择题部分 1.()部门负责日常监督检查工作,安全巡视的同时进行消防检查,推动消防安全制度的贯彻落实。 A: 消防...
    skystarwuwei阅读 15,157评论 0 3
  • 转载自机器学习中关于判断函数凸或凹以及最优化的问题感谢原博主! 在很多机器学习算法中,都会遇到最优化问题。因为我们...
    程序猪小羊阅读 2,064评论 0 1
  • 2016年新年伊始,杭州市教育局联合杭州市家庭教育学会以及杭州市家庭教育指导中心开展了“中国年”“一封...
    大小仔的幸福生活阅读 569评论 1 4