Java使用反射和动态代理实现一个View注解绑定库

 更新时间:2022年05月10日 15:54:33   作者:木水Code  
这篇文章主要介绍了Java使用反射和动态代理实现一个View注解绑定库,代码简洁,使用简单,扩展性强,结合实例代码给大家介绍的非常详细,需要的朋友可以参考下

使用反射结合动态代理实现一个View注解绑定库,支持View和事件绑定,代码简洁,使用简单,扩展性强。

支持的功能

  • @ContentView 绑定layout 替代setContentView()
  • @BindView 绑定View 替代findViewById()
  • @OnClick 绑定点击事件 替代setOnClickListener()
  • @OnLongClick 绑定长按事件 替代setOnLongClickListener()

代码

注解类

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface ContentView {
    int value();
}
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface BindView {
    int value();
}
@Target(ElementType.ANNOTATION_TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface OnEvent {
    //订阅方式
    String setCommonListener();
    //事件源对象
    Class<?> commonListener();
}
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@OnEvent(setCommonListener = "setOnClickListener",
        commonListener = View.OnClickListener.class)
public @interface OnClick {
    int value();
}
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@OnEvent(setCommonListener = "setOnLongClickListener",
        commonListener = View.OnLongClickListener.class)
public @interface OnLongClick {
    int value();
}

实现类

public class MsInjector {
    public static void inject(Object object) {
        injectContentView(object);
        injectView(object);
        injectEvent(object);
    }
    private static void injectContentView(Object object) {
        Class<?> clazz = object.getClass();
        //获取到ContentView注解
        ContentView contentView = clazz.getAnnotation(ContentView.class);
        if (contentView == null) {
            return;
        }
        //获取到注解的值,也就是layoutResID
        int layoutResID = contentView.value();
        try {
            //反射出setContentView方法并调用
            Method method = clazz.getMethod("setContentView", int.class);
            method.invoke(object, layoutResID);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    private static void injectView(Object object) {
        Class<?> clazz = object.getClass();
        //获取到所有字段并遍历
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            field.setAccessible(true);
            //获取字段上的BindView注解
            BindView bindView = field.getAnnotation(BindView.class);
            if (bindView == null) {
                continue;
            }
            //获取到viewId
            int viewId = bindView.value();
            try {
                //通过反射调用findViewById得到view实例对象
                Method method = clazz.getMethod("findViewById", int.class);
                Object view = method.invoke(object, viewId);
                //赋值给注解标注的对应字段
                field.set(object, view);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    private static void injectEvent(Object object) {
        Class<?> clazz = object.getClass();
        //获取到当前页年所有方法并遍历
        Method[] declaredMethods = clazz.getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            declaredMethod.setAccessible(true);
            //获取方法上的所有注解并遍历
            Annotation[] annotations = declaredMethod.getDeclaredAnnotations();
            for (Annotation annotation : annotations) {
                //获取注解本身
                Class<? extends Annotation> annotationType = annotation.annotationType();
                //获取注解上的OnEvent注解
                OnEvent onEvent = annotationType.getAnnotation(OnEvent.class);
                if (onEvent == null) {
                    continue;
                }
                //拿到注解中的元素
                String setCommonListener = onEvent.setCommonListener();
                Class<?> commonListener = onEvent.commonListener();
                try {
                    //由于上边没有明确获取是哪个注解,所以这里需要使用反射获取viewId
                    Method valueMethod = annotationType.getDeclaredMethod("value");
                    valueMethod.setAccessible(true);
                    int viewId = (int) valueMethod.invoke(annotation);
                    //通过反射findViewById获取到对应的view
                    Method findViewByIdMethod = clazz.getMethod("findViewById", int.class);
                    Object view = findViewByIdMethod.invoke(object, viewId);
                    //通过反射获取到view中对应的setCommonListener方法
                    Method viewMethod = view.getClass().getMethod(setCommonListener, commonListener);
                    //使用动态代理监听回调
                    Object proxy = Proxy.newProxyInstance(
                            clazz.getClassLoader(),
                            new Class[]{commonListener},
                            new InvocationHandler() {
                                @Override
                                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                                    //最终执行被标注的方法
                                    return declaredMethod.invoke(object, null);
                                }
                            }
                    );
                    //调用view的setCommonListener方法
                    viewMethod.invoke(view, proxy);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
复制代码

使用

@ContentView(R.layout.activity_main)
public class MainActivity extends AppCompatActivity {
    @BindView(R.id.button1)
    private Button button1;
    @BindView(R.id.button2)
    Button button2;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        MsInjector.inject(this);
    }
    @OnClick(R.id.button1)
    public void clickButton1() {
        Toast.makeText(this, "click button1", Toast.LENGTH_SHORT).show();
    }
    @OnClick(R.id.button2)
    public void clickButton2() {
        Toast.makeText(this, "click button2", Toast.LENGTH_SHORT).show();
    }
    @OnLongClick(R.id.button1)
    public boolean longClickButton1() {
        Toast.makeText(this, "long click button1", Toast.LENGTH_SHORT).show();
        return false;
    }
    @OnLongClick(R.id.button2)
    public boolean longClickButton2() {
        Toast.makeText(this, "long click button2", Toast.LENGTH_SHORT).show();
        return false;
    }
}

到此这篇关于Java使用反射和动态代理实现一个View注解绑定库的文章就介绍到这了,更多相关View注解绑定库内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot集成P6Spy实现SQL日志的记录详解

    SpringBoot集成P6Spy实现SQL日志的记录详解

    P6Spy是一个框架,它可以无缝地拦截和记录数据库活动,而无需更改现有应用程序的代码。一般我们使用的比较多的是使用p6spy打印我们最后执行的sql语句
    2022-11-11
  • SpringBoot自定义启动器Starter流程详解

    SpringBoot自定义启动器Starter流程详解

    SpringBoot中的starter是一种非常重要的机制,能够抛弃以前繁杂的配置,将其统一集成进starter,应用者只需要在maven中引入starter依赖,SpringBoot就能自动扫描到要加载的信息并启动相应的默认配置。starter让我们摆脱了各种依赖库的处理,需要配置各种信息的困扰
    2022-11-11
  • springboot 注册服务注册中心(zk)的两种方式详解

    springboot 注册服务注册中心(zk)的两种方式详解

    本文通过一个demo讲述一下这两种注册方式,使用的是传统的向zk注册的方案。对springboot 注册zk的相关知识感兴趣的朋友一起看看吧
    2018-01-01
  • Spring Cloud出现Options Forbidden 403问题解决方法

    Spring Cloud出现Options Forbidden 403问题解决方法

    本篇文章主要介绍了Spring Cloud出现Options Forbidden 403问题解决方法,具有一定的参考价值,有兴趣的可以了解一下
    2017-11-11
  • java 操作gis geometry类型数据方式

    java 操作gis geometry类型数据方式

    这篇文章主要介绍了java 操作gis geometry类型数据方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • 基于TCP通信丢包原因总结(推荐)

    基于TCP通信丢包原因总结(推荐)

    下面小编就为大家带来一篇基于TCP通信丢包原因总结(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • SSH原理及两种登录方法图文详解

    SSH原理及两种登录方法图文详解

    SSH(Secure Shell)是一套协议标准,可以用来实现两台机器之间的安全登录以及安全的数据传送,其保证数据安全的原理是非对称加密。本文通过图文并茂的形式给大家介绍了SSH原理及两种登录方法,一起看看吧
    2018-08-08
  • Java中return的用法(两种)

    Java中return的用法(两种)

    这篇文章主要介绍了Java中return的用法(两种)的相关资料,需要的朋友可以参考下
    2016-01-01
  • Java调用Shell命令和脚本的实现

    Java调用Shell命令和脚本的实现

    这篇文章主要介绍了Java调用Shell命令和脚本的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-02-02
  • JAVA--HashMap热门面试题

    JAVA--HashMap热门面试题

    这篇文章主要介绍了JAVA关于HashMap容易被提问的面试题,文中题目提问频率高,相信对你的面试有一定帮助,想要入职JAVA的朋友可以了解下
    2020-06-06

最新评论