viewBinding与反射结合的实用实践

news2025/1/12 16:05:41

首先,官方教程指个路:视图绑定

本文阅读认真阅读大约需要5-20分钟

也可直接跳到文末3.0看最终方案

目录

    • 1 关于ViewBinding
      • 第一点,命名符合一定的规则
      • 第二点,继承自ViewBinding
    • 2 正常使用不便之处
    • 3 话不多说直接看代码1.0
    • 4 话不多说直接看代码2.0
    • 5 话不多说直接看代码3.0

1 关于ViewBinding

XXXBinding是自动生成的:
1、命名符合一定的规则
2、所有的XXXBinding都继承自ViewBinding基类

第一点,命名符合一定的规则

这个在官方文档可以看到:

为某个模块启用视图绑定功能后,系统会为该模块中包含的每个 XML 布局文件生成一个绑定类。每个绑定类均包含对根视图以及具有 ID 的所有视图的引用。系统会通过以下方式生成绑定类的名称:将 XML 文件的名称转换为驼峰式大小写,并在末尾添加“Binding”一词。

例如,Activity的命名是FirstTestActivity,那么生成的对应binding类名为ActivityTestFirstBinding

第二点,继承自ViewBinding

这个在代码里可以看出来:
在这里插入图片描述

2 正常使用不便之处

按照官方文档,在Activity中使用如下:

    private lateinit var binding: ResultProfileBinding

    override fun onCreate(savedInstanceState: Bundle) {
        super.onCreate(savedInstanceState)
        binding = ResultProfileBinding.inflate(layoutInflater)
        val view = binding.root
        setContentView(view)
    }
    

注意,这里定义了一个lateinit var变量
这意味着我们在使用前必须手动初始化
正如官方文档,在Activity#onCreate()方法中,我们需要手动调用inflate方法进行初始化
显然这其中有些代码是固定模板的:

binding = ResultProfileBinding.inflate(layoutInflater)  // ResultProfileBinding只有这个Binding类名会变化
val view = binding.root
setContentView(view)

换一个YyyActivity相应的模板只有YyyBinding不一样,其他都不变

同样,我们看下在Fragment中是怎样的:

    private var _binding: ResultProfileBinding? = null
    // This property is only valid between onCreateView and
    // onDestroyView.
    private val binding get() = _binding!!

    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        _binding = ResultProfileBinding.inflate(inflater, container, false)
        val view = binding.root
        return view
    }

    override fun onDestroyView() {
        super.onDestroyView()
        _binding = null
    }

可以看出,Fragment中使用viewBinding也是如此,存在模板代码

至此你应该知道我想做什么了:能不能将模板代码去掉?

答案是肯定的,我给出的实践方案是:反射+泛型

3 话不多说直接看代码1.0

我们可以直接利用上述的Binding命名的规则来做

/**
 * 通过泛型+反射的方式简化ViewBinding的初始化,化繁为简,不用每次再手动初始化,只需要传入类型即可
 * 继承规则:子类后缀必须以Activity结尾
 */
open class BaseActivity<BindingClass> : AppCompatActivity() {
    //私有化的Binding类,类型即为实际使用的Binding类型
    private var innerBinding: BindingClass? = null
    //暴露给子类的Binding类,返回innerBinding的非空类型主要是为了避免子类使用
    //的时候需要频繁加上!!(如有更好的解决方法请务必告诉我)
    val binding: BindingClass by lazy { innerBinding!! }
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        //以下操作基本相同
        val acName = javaClass.simpleName
        val name = acName.substring(0, acName.indexOf("Activity"))
        val bindingClass =
            classLoader.loadClass("com.example.test.databinding.Activity${name}Binding")
        //最后强转为以泛型传入的实际Binding的类型
        innerBinding = bindingClass.getMethod("inflate", LayoutInflater::class.java)
            .invoke(null, layoutInflater) as BindingClass
    }
}

// 子类
class MainActivity : BaseActivity<ActivityMainBinding>() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        //关联布局(直接使用BaseActivity的binding即可)
        setContentView(binding.root)
    }
}

到这里,子类只需要再关联布局setContentView(binding.root)即可

4 话不多说直接看代码2.0

实际上,细心的朋友会发现,1.0版本是有限制的:
定义的XxxActivity必须以Activity结尾,并且所在的包固定com.example.test.databinding

这相当于死的代码,这是不提倡的,也是不好的
那么我们继续想办法解决这个问题

我们现在实际上是需要解决这个问题loadClass("com.example.test.databinding.Activity${name}Binding")

再注意,上述1.0我们使用到了泛型,在子类中我们实际给出了Binding的类名

class MainActivity : BaseActivity<ActivityMainBinding>()

这不就是泛型参数的实参名吗?

因此,对反射熟悉的朋友应该可以想到,我们下一步就是要通过反射clazz.getGenericSuperclass().getActualTypeArguments(),将这个子类泛型实参的类名拿到

话不多说,直接看代码:

//BaseActivity.kt
open class BaseActivity<BindingClass> : AppCompatActivity() {
    //私有化的Binding类,类型即为实际使用的Binding类型
    private var innerViewBinding: BindingClass? = null
    // 通过lazy的方式,避免在创建是初始化发生错误。因为实际上官方模板的用法,binding需要再onCreate之后初始化 LayoutInflater
    // 那么这里通过lazy的方式,后续子类不再需要手动初始化
    protected val mViewBinding: BindingClass by lazy { innerViewBinding!! }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // 通过反射的方法拿到对应视图的binding类名和类
        val actualGenericTypeName = GenericUtil.getActualGenericTypeName(this.javaClass)
        MyLogUtil.d("binding", "this.javaClass...: $actualGenericTypeName")
        // 类加载器加载类
        val bindingClass = classLoader.loadClass(actualGenericTypeName)
        // 正常一个Activity 中 viewBinding的初始化:
        // binding = ResultProfileBinding.inflate(layoutInflater)
        // Fragment中:
        // _binding = ResultProfileBinding.inflate(inflater, container, false)
        innerViewBinding = bindingClass
            .getMethod("inflate", LayoutInflater::class.java)
            .invoke(null, layoutInflater) as BindingClass
        // inflate方法是ViewBinding的类方法,不是对象方法,因此obj参数为null

    }
}
// GenericUtil.java
public class GenericUtil {
    public static String getActualGenericTypeName(@NonNull Class clazz) {
        String className;
        Type genericSuperclass = clazz.getGenericSuperclass();
        ParameterizedType parameterizedType = (ParameterizedType) genericSuperclass;
        try {
            Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
            Class tClass = (Class) actualTypeArguments[0];
            className = tClass.getName();
        } catch (Exception e) {
            throw new IllegalArgumentException("Wrong ViewBinding Type");
        }
        return className;
    }
}

使用:

// 子类
class MainActivity : BaseActivity<ActivityMainBinding>() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        //关联布局(直接使用BaseActivity的binding即可)
        setContentView(binding.root)
    }
}

5 话不多说直接看代码3.0

到此,基本解决了

但是有朋友还会发现,子类中setContentView(binding.root)这行代码也是固定的模板代码,再极致点能否把这个也简化掉呢

答案当然是肯定的

直接操作:

binding在BaseActivity中已经完成加载了,那直接把setContentView(binding.root)写到BaseActivity中行不行?
NO,实际操作就发现不用编译,编译器就提示error了:
在这里插入图片描述
为什么?请朋友自己思考一下动态加载过程

然后你会发现,这个问题可以继续通过动态加载完成:

// BaseActivity.kt
val invoke = bindingClass.getMethod("getRoot").invoke(mViewBinding) as View
setContentView(invoke)

完整代码:

// BaseActivity.kt
open class BaseActivity<BindingClass> : AppCompatActivity() {
    //私有化的Binding类,类型即为实际使用的Binding类型
    private var innerViewBinding: BindingClass? = null
    protected val mViewBinding: BindingClass by lazy { innerViewBinding!! }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // 通过反射的方法拿到对应视图的binding类名和类
        val actualGenericTypeName = GenericUtil.getActualGenericTypeName(this.javaClass)
        val bindingClass = classLoader.loadClass(actualGenericTypeName)
        innerViewBinding = bindingClass
            .getMethod("inflate", LayoutInflater::class.java)
            .invoke(null, layoutInflater) as BindingClass
        val invoke = bindingClass.getMethod("getRoot").invoke(mViewBinding) as View
        setContentView(invoke)
    }
}

在子类中使用:

class MainActivity : BaseActivity<ActivityMainBinding>() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // ...完成子类其他逻辑...
    }
}

子类根本不需要干啥事,只需要把泛型参数Binding给即可!

大功告成,实际使用上来方便许多

以上就是本次实践,对你有帮助的话动一下你发财的小手点个赞关注~

有其他见解欢迎私信或评论区交流~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1178935.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据结构与算法之美学习笔记:16 | 二分查找(下):如何快速定位IP对应的省份地址?

目录 前言二分查找的变形问题变体一&#xff1a;查找第一个值等于给定值的元素变体二&#xff1a;查找最后一个值等于给定值的元素变体三&#xff1a;查找第一个大于等于给定值的元素变体四&#xff1a;查找最后一个小于等于给定值的元素 解答开篇内容小结 前言 本节课程思维导…

vue3+ts 项目遇到的问题和bug

1.router中使用pinia报错 pinia.mjs:1709 Uncaught Error: [&#x1f34d;]: "getActivePinia()" was called but there was no active Pinia. Are you trying to use a store before calling "app.use(pinia)"? See https://pinia.vuejs.org/core-concep…

[科研图像处理]用matlab平替image-j,有点麻烦,但很灵活!

做材料与生物相关方向的同学应该对image-j并不陌生&#xff0c;前几天有个师兄拜托我用image-j分析一些图片&#xff0c;但使用过后发现我由于不了解image-j的工作流程而对结果并不确信&#xff0c;而且image-j的功能无法拓展&#xff0c;对有些图片的处理效果并不好&#xff0…

【网络协议】聊聊HTTPDNS如何工作的

传统 DNS 存在哪些问题&#xff1f; 域名缓存问题 我们知道CND会进行域名解析&#xff0c;但是由于本地会进行缓存对应的域名-ip地址&#xff0c;所以可能出现过期数据的情况。 域名转发问题 出口 NAT 问题 域名更新问题 解析延迟问题 因为在解析DNS的时候&#xff0c;需要进行…

【C语言】函数的系统化精讲(二)

&#x1f308;write in front :&#x1f50d;个人主页 &#xff1a; 啊森要自信的主页 &#x1f308;作者寄语 &#x1f308;&#xff1a; 小菜鸟的力量不在于它的体型&#xff0c;而在于它内心的勇气和无限的潜能&#xff0c;只要你有决心&#xff0c;就没有什么事情是不可能的…

vue3错误排查-POST请求的body参数 传参方式form-data和json

问题&#xff1a;vue3实现登录功能&#xff0c;登录成功后 跳转到登陆后的界面 一秒后 闪退回登录页 对应的输出结果也一闪而过&#xff0c;反复复查了代码&#xff0c;没问题。 自测&#xff1a;进行断点输出调试。强行跳转到登陆后的界面&#xff0c;查看输出的结果。 没有报…

安全操作(安卓推流)程序

★ 安全操作项目 项目描述&#xff1a;安全操作项目旨在提高医疗设备的安全性&#xff0c;特别是在医生离开操作屏幕时&#xff0c;以减少非授权人员的误操作风险。为实现这一目标&#xff0c;我们采用多层次的保护措施&#xff0c;包括人脸识别、姿势检测以及二维码识别等技术…

Linux 实现原理 — NUMA 多核架构中的多线程调度开销与性能优化

前言 NOTE&#xff1a;本文中所指 “线程” 均为可执行调度单元 Kernel Thread。 NUMA 体系结构 NUMA&#xff08;Non-Uniform Memory Access&#xff0c;非一致性存储器访问&#xff09;的设计理念是将 CPU 和 Main Memory 进行分区自治&#xff08;Local NUMA node&#x…

1070 结绳

一.问题&#xff1a; 给定一段一段的绳子&#xff0c;你需要把它们串成一条绳。每次串连的时候&#xff0c;是把两段绳子对折&#xff0c;再如下图所示套接在一起。这样得到的绳子又被当成是另一段绳子&#xff0c;可以再次对折去跟另一段绳子串连。每次串连后&#xff0c;原来…

【网络安全 --- web服务器解析漏洞】IIS,Apache,Nginx中间件常见解析漏洞

一&#xff0c;工具及环境准备 以下都是超详细保姆级安装教程&#xff0c;缺什么安装什么即可&#xff08;提供镜像工具资源&#xff09; 1-1 VMware 16.0 安装 【网络安全 --- 工具安装】VMware 16.0 详细安装过程&#xff08;提供资源&#xff09;-CSDN博客文章浏览阅读20…

Android Datastore 动态创建与源码解析

涉及到的知识点 1、协程原理---->很好的博客介绍&#xff0c;一个小故事讲明白进程、线程、Kotlin 协程到底啥关系&#xff1f; 2、Channel知识点---->Android—kotlin-Channel超详细讲解 3、Coroutines : CompletableDeferred and structured concurrency 封装的DataS…

动态规划-丑数

** 描述 把只包含质因子2、3和5的数称作丑数&#xff08;Ugly Number&#xff09;。例如6、8都是丑数&#xff0c;但14不是&#xff0c;因为它包含质因子7。 习惯上我们把1当做是第一个丑数。求按从小到大的顺序的第 n个丑数。 数据范围&#xff1a; 0≤n≤2000 要求&#x…

Mysql数据库 9.SQL语言 查询语句 连接查询、子查询

连接查询 通过查询多张表&#xff0c;用连接查询进行多表联合查询 关键字&#xff1a;inner join 内连接 left join 左连接 right join 右连接 数据准备 创建新的数据库&#xff1a;create database 数据库名; create database db_test2; 使用数据库&#xff1a;use 数据…

企业如何搭建智能客服系统?

在数字化时代&#xff0c;企业面临着客户需求多样化、市场竞争日益激烈等多重挑战。为了更好地满足客户的需求、提供高效的服务&#xff0c;越来越多的企业开始搭建智能客服系统。智能客服系统结合了人工智能和自然语言处理技术&#xff0c;可以实现自动回复、智能推荐以及数据…

Linux内核移植之主频设置

一. Linux内核移植 正点原子 ALPHA开发板已经添加到 Linux内核里面去了&#xff0c;前面文章关于如何添加已经掌握。但是&#xff0c;还有一些驱动的问题需要修改。 正点原子 I.MX6U-ALPHA 开发板所使用的 I.MX6ULL 芯片主频都是 792MHz 的&#xff0c;也就是NXP 官方宣…

MySQL的event的使用方法

MySQL的event的使用方法 一、事件定时策略 1、查看event事件开启状态 SHOW VARIABLES LIKE event_scheduler;如图&#xff0c;Value值 ON&#xff1a;打开&#xff0c;OFF&#xff1a;关闭。 2、设置event事件打开 SET GLOBAL event_scheduler ON;如果MySQL重启了&#x…

Python模块导入出现ModuleNotFoundError: No module named ‘***’解决方法

概述 几年没弄python了&#xff0c;全部还会给老师&#xff0c;今天弄了个demo&#xff0c;老是报错&#xff0c;在此记录下&#xff0c;方便后续查阅。 环境&#xff1a;Windows10 开发IDEA&#xff1a;PyCharm 2023.1.3 1、报错如下所示 2、解决方法&#xff1a;安装execjs…

AJAX-解决回调函数地狱问题

一、同步代码和异步代码 1.同步代码 浏览器是按照我们书写代码的顺序一行一行地执行程序的。浏览器会等待代码的解析和工作&#xff0c;在上一行完成之后才会执行下一行。这也使得它成为一个同步程序。 总结来说&#xff1a;逐行执行&#xff0c;需原地等待结果后&#xff0…

[python 刷题] 437 Path Sum III

[python 刷题] 437 Path Sum III 之前有写过 Path Sum I & II, leetcode 112 & 113&#xff0c;虽然使用 JS 写的&#xff0c;不过 python 的实现也更新了一下 题目如下&#xff1a; Given the root of a binary tree and an integer targetSum, return the number o…

解释一下Node.js中的事件循环(event loop)

聚沙成塔每天进步一点点 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 欢迎来到前端入门之旅&#xff01;感兴趣的可以订阅本专栏哦&#xff01;这个专栏是为那些对Web开发感兴趣、刚刚踏入前端领域的朋友们量身打造的。无论你是完全的新手还是有一些基础的开发…