Databinding (一) 基本用法

2024-02-28 08:08
文章标签 用法 基本 databinding

本文主要是介绍Databinding (一) 基本用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述
2015 年的 I/O 大会上,Google 发布了官方的数据绑定框架 Data Binding Library,使得官方原生支持 MVVM。

Data Binding 是把数据直接绑定到 XML 文件上,并能实现自动刷新。

Data Binding 减少了代码的耦合性,一些如 findViewById、setText 之类的操作都可以通过绑定实现。

环境配置
环境配置非常简单,只要在 build.gradle 文件里面定义一下代码即可

android {....dataBinding {enabled = true}
}


一个简单的例子
1、首先,定义一个 Java Bean

public class User {private String name;private int age;public User(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}public void setName(String name) {this.name = name;}public void setAge(int age) {this.age = age;}
}


2、然后定义 Layout 文件

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"><data><variablename="user"type="com.example.tianjf.myapplication.User" /></data><LinearLayoutandroid:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@{user.name}" /><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@{String.valueOf(user.age)}" /></LinearLayout>
</layout>


Layout 文件和之前有什么不同呢?

首先,在最外层再套一层 标签。

然后,在 layout 标签下面定义 标签。 
data 标签下面的 variable 定义数据绑定用的实体类。这个实体是从外部传进来的(具体怎么传下文再讲)。 
type 里面是完整的带包名的类, 
name 自定义一个名称,下面具体绑定的时候就是用的这个名称。

最后,用 @{} 来把数据绑定到 UI 上。 
@{user.name} 把 name 属性绑定到第一个 TextView 上。 
@{String.valueOf(user.age)} 把 age 属性绑定到第二个 TextView 上。这里 age 是 int 类型的,所以需要把它转化成 String 类型。由于 String 是属于 java.lang 下面的,所以不需要 import。java.lang 以外的类是需要 import 的(具体怎么 import 下文再讲)。

3、数据绑定 
之前的 Layout 文件的定义会默认生成一个数据绑定类,这个数据绑定类的名称默认和 Layout 文件的类名有关。比如 activity_main.xml 会生成 ActivityMainBinding.java 文件。

我们来看看代码

public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);binding.setUser(new User("Jack", 10));}
}


用 DatabindingUtil.setContentView() 替换之前的 setContentView(),返回值是自动生成的 ActivityMainBinding,然后调用 setUser 方法把 User 实例绑定到 XML 文件中去。

这样,运行之后就可以看到 User 的信息被显示到了画面上了。

生成的 Binding 类的获取方式
上面例子中由于是 Activity 的 Layout 文件,所以使用了 DataBindingUtil.setContentView 来获取。

除了上面的方法,还可以通过 inflate 获取。

MainActivityBinding binding = MainActivityBinding.inflate(getLayoutInflater());


如果是在 ListView 或者 RecyclerView 的 adapter 中

ListItemBinding binding = ListItemBinding.inflate(layoutInflater, viewGroup, false);
//or
ListItemBinding binding = DataBindingUtil.inflate(layoutInflater, R.layout.list_item, viewGroup, false);


控制自动生成类的生成方式
默认情况下,会根据 XML 文件的名称(具体规则上文已经提及)在 .databinding 目录下生成文件。比如 package 名为 com.example.myapplication,那么会在 com.example.myapplication.databinding 下面生成文件。

生成的文件的名称也可以自定义。

<data class="MainBinding">...
</data>


这样的话,就会在 com.example.myapplication.databinding 下面生成 MainBinding 的文件。

生成的文件的路径也可以自定义。 
如果想直接在 package 下面生成

<data class=".MainBinding">...
</data>


加一个 . 就会在 com.example.myapplication 下面生成 MainBinding 的文件。

当然,不想使用 package 名,想自己自定义路径,也是可以的,写全你想要的路径即可

<data class="com.example.MainBinding">...
</data>


import
当在 XML 中数据绑定的时候,用到了 java.lang 之外的类,必须在 data 标签下面 import。 
比如想控制 View 的显示和隐藏。

<TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@{user.name}"android:visibility="@{user.showName ? View.VISIBLE : View.GONE}" />


由于用到了 View 类,所以必须 import。

<data><import type="android.view.View" />......
</data>


variable 中用到的类也可以先 import 在使用,其实和 Java 是一样的。

<data><import type="com.example.tianjf.myapplication.User" /><variablename="user"type="User" />
</data>


如果类名相同,package 名不相同,上面的写法就会出现 type 不知道指定哪个类。但是别担心,可以用别名解决。

<data><import type="com.example.tianjf.myapplication.User" alias="User1" /><import type="com.example.tianjf.User" alias="User2" /><variablename="user1"type="User1" /><variablename="user2"type="User2" />
</data>


import 的类型可以用到 variable 中

    <data><import type="com.example.tianjf.myapplication.User" /><import type="java.util.List"/><variable name="user" type="User"/><variable name="userList" type="List<User>"/></data>


import 的类型可以用到表达式中

<TextViewandroid:text="@{((User)(user.connection)).lastName}"android:layout_width="wrap_content"android:layout_height="wrap_content"/>


import 的类型可以调用它的 static 变量和 static 方法

<data><import type="com.example.MyStringUtils"/><variable name="user" type="com.example.User"/>
</data>
…
<TextViewandroid:text="@{MyStringUtils.capitalize(user.lastName)}"android:layout_width="wrap_content"android:layout_height="wrap_content"/>


include
当 layout 文件中用到 include 的时候,variable 也可以传到 include 的 layout 文件中继续使用

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:bind="http://schemas.android.com/apk/res-auto"><data><variable name="user" type="com.example.User"/></data><LinearLayoutandroid:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><include layout="@layout/name"bind:user="@{user}"/><include layout="@layout/contact"bind:user="@{user}"/></LinearLayout>
</layout>


Null Safe
DataBinding 是 Null Safe 的,比如下列代码

<TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@{user.name}"


如果 user 为 null,@{user.name} 也将为 null,并不会出现 NullPointerException。

表达式的使用
DataBinding 的时候可以指定绑定传进来的值,也可以使用表达式达到各种效果!

前面提到的控制 View 的显示和隐藏

<TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@{user.name}"android:visibility="@{user.showName ? View.VISIBLE : View.GONE}" />


?? 操作符

android:text="@{user.displayName ?? user.lastName}"


如果不为 null,则选取 ?? 左边的,如果为 null,则选取 ?? 右边的,相当于以下代码

android:text="@{user.displayName != null ? user.displayName : user.lastName}"


集合的使用

<data><import type="android.util.SparseArray"/><import type="java.util.Map"/><import type="java.util.List"/><variable name="list" type="List<String>"/><variable name="sparse" type="SparseArray<String>"/><variable name="map" type="Map<String, String>"/>
</data>
…
android:text="@{list[0]}"
…
android:text="@{sparse[0]}"
…
android:text="@{map['key']}"


Resources 的使用

DataBinding 的时候也可以使用 resources

android:padding="@{large? @dimen/largePadding : @dimen/smallPadding}"


Format strings 也可以使用。 
比如下面的 string

<string name="name">My name is %s</string>


可以使用 String.format 传入参数

android:text="@{String.format(@string/name,user.name)}"


也可以这样写

android:text="@{@string/name(user.name)}"


双向绑定
前面介绍的都是单向绑定,即 ViewModel 的值绑定到 UI 上。如果希望 UI 的变更也反应到 ViewModel,那么就需要双向绑定了。其实双向绑定很简单,只需要加个 = 就好了。

<EditTextandroid:layout_width="match_parent"android:layout_height="wrap_content"android:text="@={viewModel.firstName}"/>

 

这篇关于Databinding (一) 基本用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/754901

相关文章

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

前端知识点之Javascript选择输入框confirm用法

《前端知识点之Javascript选择输入框confirm用法》:本文主要介绍JavaScript中的confirm方法的基本用法、功能特点、注意事项及常见用途,文中通过代码介绍的非常详细,对大家... 目录1. 基本用法2. 功能特点①阻塞行为:confirm 对话框会阻塞脚本的执行,直到用户作出选择。②

Python中多线程和多进程的基本用法详解

《Python中多线程和多进程的基本用法详解》这篇文章介绍了Python中多线程和多进程的相关知识,包括并发编程的优势,多线程和多进程的概念、适用场景、示例代码,线程池和进程池的使用,以及如何选择合适... 目录引言一、并发编程的主要优势二、python的多线程(Threading)1. 什么是多线程?2.

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

MyBatis-Flex BaseMapper的接口基本用法小结

《MyBatis-FlexBaseMapper的接口基本用法小结》本文主要介绍了MyBatis-FlexBaseMapper的接口基本用法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具... 目录MyBATis-Flex简单介绍特性基础方法INSERT① insert② insertSelec

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

深入解析Spring TransactionTemplate 高级用法(示例代码)

《深入解析SpringTransactionTemplate高级用法(示例代码)》TransactionTemplate是Spring框架中一个强大的工具,它允许开发者以编程方式控制事务,通过... 目录1. TransactionTemplate 的核心概念2. 核心接口和类3. TransactionT

数据库使用之union、union all、各种join的用法区别解析

《数据库使用之union、unionall、各种join的用法区别解析》:本文主要介绍SQL中的Union和UnionAll的区别,包括去重与否以及使用时的注意事项,还详细解释了Join关键字,... 目录一、Union 和Union All1、区别:2、注意点:3、具体举例二、Join关键字的区别&php

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(