注解+反射+动态代理实现View点击事件绑定

一、一些感想

其实在工作的过程中,我一直感觉自己的java基础还是很薄弱的,所以不得不重新看看java基础,其实注解在Android应用实在很广泛,它让代码简介,并且解耦,提高了很多开发效率。为了巩固对基础知识的理解,所以干脆使用注解+反射+动态代理实现View的点击事件绑定功能,加深印象。在整个实现过程中要求的知识点还是比较多,首先要熟悉Android View点击事件,当然如果对注解不了解,那也就没法理解注解的注解,最后也就是动态代理了,不过理解了也就运用自如了。

二、具体实现

1,定义View 事件注解类型,在Android 中View的点击事件分为点击和长按两种,定义事件类型也是为后续工作做好铺点:

/**
 * Description:点击事件类型定义 事件分为长按和短按(即点击) 
 * ElementType.ANNOTATION_TYPE对注解进行注解<br>
 * Author:Frank<br>
 * Date:2020/5/6<br>
 * Version:V1.0.0<br>
 * Update: <br>
 */
@Target(ElementType.ANNOTATION_TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface EventType {
    Class listenerType();
    String listenerSetter();
}
2,定义点击事件
/**
 * Description:短按事件类型(点击)<br>
 * Author:Frank<br>
 * Date:2020/5/6<br>
 * Version:V1.0.0<br>
 * Update: <br>
 */
@EventType(listenerType = View.OnClickListener.class, listenerSetter = "setOnClickListener")
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface OnClick {
    @IdRes int[] myValue();
}

3,定义长按事件
/**
 * Description:长按事件类型<br>
 * Author:Frank<br>
 * Date:2020/5/6<br>
 * Version:V1.0.0<br>
 * Update: <br>
 */
@EventType(listenerType = View.OnLongClickListener.class, listenerSetter = "setOnLongClickListener")
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface OnLongClick {
    @IdRes int[] myValue();
}

4,获取注解,使用反射和动态代理完成事件绑定
/**
 * Description:使用反射和动态代理实现View点击事件绑定<br>
 * Author:Frank<br>
 * Date:2020/5/6<br>
 * Version:V1.0.0<br>
 * Update: <br>
 */
public class InjectViewClickUtil {
    private static final String TAG = InjectViewClickUtil.class.getSimpleName();

    public static void inject(Activity activity) {
        if (activity == null) {
            return;
        }
        //得到类对象
        Class<? extends Activity> clz = activity.getClass();
        //获取类对象中的所有方法
        Method[] methods = clz.getDeclaredMethods();
        //判断是否存在成员方法
        if (methods == null) {
            return;
        }
        //遍历方法
        for (Method m : methods) {
            //获取方法上的所有注解
            Annotation[] annotations = m.getAnnotations();
            //判断是否存在注解
            if (annotations == null) {
                continue;
            }
            //遍历所有注解
            for (Annotation a : annotations) {
                //得到注解类型对象
                Class<? extends Annotation> annotationType = a.annotationType();
                if (annotationType.isAnnotationPresent(EventType.class)) {
                    //得到具体注解对象
                    EventType eventType = annotationType.getAnnotation(EventType.class);
                    //取值
                    Class listenerType = eventType.listenerType();
                    Log.d(TAG, "inject: " + listenerType);
                    String listenerSetter = eventType.listenerSetter();
                    try {
                        //得到标记有OnClick 和 OnLongClick 注解的方法
                        Method valueMethod = annotationType.getDeclaredMethod("myValue");
                        //得到所有需要点击控件的id 也就是注解value
                        int[] ids = (int[]) valueMethod.invoke(a);
                        //设置权限
                        m.setAccessible(true);
                        //InvocationHandler接口是proxy代理实例的调用处理程序实现的一个接口,
                        // 每一个proxy代理实例都有一个关联的调用处理程序;在代理实例调用方法时,方法调用被编码分派到调用处理程序的invoke方法
                        ListenerInvocationHandler invocationHandler = new ListenerInvocationHandler(m, activity);
                        //创建代理对象 最终会回调我们定义注解的方法
                        Object proxyInstance = Proxy.newProxyInstance(clz.getClassLoader(), new Class[]{listenerType}, invocationHandler);
                        for (int id : ids) {
                            View view = activity.findViewById(id);
                            Method setter = view.getClass().getMethod(listenerSetter, listenerType);
                            setter.invoke(view, proxyInstance);
                        }
                    } catch (NoSuchMethodException e) {
                        e.printStackTrace();
                    } catch (IllegalAccessException e) {
                        e.printStackTrace();
                    } catch (InvocationTargetException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }

    static class ListenerInvocationHandler<T> implements InvocationHandler {
        private Method method;
        private T target;

        public ListenerInvocationHandler(Method method, T target) {
            this.target = target;
            this.method = method;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            return this.method.invoke(target, args);
        }
    }
}

三、如何使用

public class MainActivity extends AppCompatActivity {
    private static final String TAG = MainActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        InjectViewClickUtil.inject(this);
    }

    @OnClick(myValue = {R.id.tv1, R.id.tv2})
    public void onClick(View v) {
        if (v.getId() == R.id.tv1) {
            Toast.makeText(this, ((TextView) v).getText(), Toast.LENGTH_SHORT).show();

        }
        if (v.getId() == R.id.tv2) {
            Toast.makeText(this, ((TextView) v).getText(), Toast.LENGTH_SHORT).show();
        }
    }

    @OnLongClick(myValue = {R.id.tv1, R.id.tv2})
    public boolean onLongClick(View view) {
        switch (view.getId()) {
            case R.id.tv1:
                Log.d(TAG, "onLongClick: 长安了textView1");
                break;
            case R.id.tv2:
                Log.d(TAG, "onLongClick: 长按了textView2");
                break;
        }
        return true;
    }
}

注意事项:Android View长按事件是需要返回值的即代表当前事件是否消费, public boolean
onLongClick(View view) { return false};由于在实现过程中没有注意这一问题,最后运行报错,检查发现我定义了一个void类型方法,导致执行的时候为空类型不对,直接崩溃。

四,总结

java动态代理机制中有两个重要的类和接口InvocationHandler(接口)和Proxy(类),这一个类Proxy和接口InvocationHandler是我们实现动态代理的核心;如果对动态代理不是很了解可以先看看这篇文章(https://blog.csdn.net/yaomingyang/article/details/80981004),由于是在运行是通过反射获取属性,所以注解选择了运行时,关于注解的生命周期可以根据实际业务自行选择。

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