Android DataBinding应用(一)
背景:最近项目中使用到了 DataBinding (Google 在 2015年开发者大会提出), 正好目前的项目中也有所涉及,现在来学习一下
https://blog.csdn.net/qq_33689414/article/details/52205703
https://blog.csdn.net/qq_33689414/article/details/52205718
https://blog.csdn.net/qq_33689414/article/details/52205724
https://blog.csdn.net/qq_33689414/article/details/52205730
https://blog.csdn.net/qq_33689414/article/details/52205734
开始学习
1 :什么是DataBinging ,DataBing解决了什么问题
- 需要多次使用findViewById,损害了应用性能且令人厌烦
- 更新UI数据需切换至UI线程,将数据分解映射到各个view比较麻烦
2:DataBinging的导入
在应用的build.gradle中添加下面代码:
android {
dataBinding {
enabled true
}
}
3: DataBinding 基本使用:
3.1 摆脱单纯的 findviewbyid
public class DataBingingActivityDemo extends AppCompatActivity {
private ActivityMainDemoBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 1: 只要通过 DataBinging 加载布局 都会对应的生成 Binding对象,比如我们生成一个 MainActivityBinding对象
// https://blog.csdn.net/huangxin388/article/details/77425678
// 注意一定要把格式搞对了,才能自动生成 ActivityMainDemoBinding 对象
binding = DataBindingUtil.setContentView(this, R.layout.activity_main_demo);
// 相当于 findViewById
binding.mainTextview.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
}
});
}
}
3.2 绑定基本数据类型及String
public class DataBingingActivityDemo extends AppCompatActivity {
private ActivityMainDemoBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
/* 只要通过 DataBinging 加载布局 都会对应的生成 Binding对象,比如我们生成一个 MainActivityBinding对象
* https://blog.csdn.net/huangxin388/article/details/77425678
* 注意一定要把格式搞对了,才能自动生成 ActivityMainDemoBinding 对象
*/
binding = DataBindingUtil.setContentView(this, R.layout.activity_main_demo);
// 相当于 findViewById
binding.mainTextview.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
}
});
/**
* 在来试验 通过操作符给控件设置一些信息
* 直接给 Button 设置文本和 控件是否能被点击 ,需要在 xml 的配置文件配置一些 操作符
*/
binding.setContent("对String类型数据的绑定");
binding.setEnabled(false);//设置enabled值为false
//给控件设置点击事件,发现其实点击无效,因为在布局文件中给cilckable属性绑定了enabled,在代码中设置enabled值为false,所以点击事件无效
binding.mainTv2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(DataBingingActivityDemo.this, "我被点击了", Toast.LENGTH_SHORT).show();
}
});
}
<?xml version="1.0" encoding="utf-8"?>
<layout
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">
<data>
<!--绑定基本数据类型及String-->
<!--name: 和java代码中的对象是类似的,名字自定义-->
<!--type: 和java代码中的类型是一致的-->
<variable
name="content"
type="String" />
<variable
name="enabled"
type="boolean" />
</data>
<android.support.constraint.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".DataBingingActivityDemo">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="你好 DataBinging,单纯的摆脱findViewById"
android:textSize="30dp"
android:id="@+id/main_textview"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<!--绑定基本数据类型及String的使用是通过 @{数据类型的对象} 通过对应数据类型的控制显示-->
<Button
android:id="@+id/main_tv2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:clickable="@{enabled}"
android:text="@{content}"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="@id/main_textview" />
</android.support.constraint.ConstraintLayout>
</layout>
3.3 绑定Model 数据
public class DataBingingActivityDemo extends AppCompatActivity {
private ActivityMainDemoBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
/* 只要通过 DataBinging 加载布局 都会对应的生成 Binding对象,比如我们生成一个 MainActivityBinding对象
* https://blog.csdn.net/huangxin388/article/details/77425678
* 注意一定要把格式搞对了,才能自动生成 ActivityMainDemoBinding 对象
*/
binding = DataBindingUtil.setContentView(this, R.layout.activity_main_demo);
// 相当于 findViewById
binding.mainTextview.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
}
});
/**
* 在来试验通过 绑定 Model 数据类型
* binding.setVariable(BR.user,user);
*/
DataModel data = new DataModel("绑定Model数据类型");
binding.setVariable(BR.dataModel,data);
}
}
<data>
<!--绑定Model数据2中形式,一中是导入该类型的,一种指定类型的全称,和java一样-->
<!-- 方式一 -->
<variable
name="dataModel"
type="com.application.databinging.model.DataModel"/>
<!-- 方式二 -->
<!--<import type="www.zhang.com.databinding.User" />-->
<!--<variable-->
<!--name="user"-->
<!--type="User" />-->
</data>
3.4 绑定事件
<?xml version="1.0" encoding="utf-8"?>
<layout
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">
<data>
<!--绑定Model数据2中形式,一中是导入该类型的,一种指定类型的全称,和java一样-->
<!-- 方式一 -->
<!-- <variable-->
<!-- name="dataModel"-->
<!-- type="com.application.databinging.model.DataModel"/>-->
<!-- 方式二 -->
<!--<import type="www.zhang.com.databinding.User" />-->
<!--<variable-->
<!--name="user"-->
<!--type="User" />-->
<!-- 方式三 -->
<!--绑定事件-->
<variable
name="event"
type="com.application.databinging.event.EventClickInterface" />
<variable
name="title1"
type="String" />
<variable
name="title2"
type="String" />
<variable
name="title3"
type="String" />
<variable
name="title4"
type="String" />
</data>
<android.support.constraint.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".DataBingingActivityDemo">
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="@{event.click1}"
android:text="@{title1}"
tools:ignore="MissingConstraints" />
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="@{event::click2}"
android:text="@{title2}"
tools:ignore="MissingConstraints" />
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="@{()->event.cilck3(title4)}"
android:text="@{title3}"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<!-- <TextView-->
<!-- android:layout_width="wrap_content"-->
<!-- android:layout_height="wrap_content"-->
<!-- android:text="你好 DataBinging,单纯的摆脱findViewById"-->
<!-- android:textSize="30dp"-->
<!-- android:id="@+id/main_textview"-->
<!-- app:layout_constraintBottom_toBottomOf="parent"-->
<!-- app:layout_constraintLeft_toLeftOf="parent"-->
<!-- app:layout_constraintRight_toRightOf="parent"-->
<!-- app:layout_constraintTop_toTopOf="parent" />-->
<!--绑定基本数据类型及String的使用是通过 @{数据类型的对象} 通过对应数据类型的控制显示-->
<!-- <Button-->
<!-- android:id="@+id/main_tv2"-->
<!-- android:layout_width="match_parent"-->
<!-- android:layout_height="wrap_content"-->
<!-- android:clickable="@{enabled}"-->
<!-- android:text="@{content}"-->
<!-- app:layout_constraintBottom_toBottomOf="parent"-->
<!-- app:layout_constraintLeft_toLeftOf="parent"-->
<!-- app:layout_constraintRight_toRightOf="parent"-->
<!-- app:layout_constraintTop_toTopOf="@id/main_textview" />-->
</android.support.constraint.ConstraintLayout>
</layout>
public class DataBingingActivityDemo extends AppCompatActivity {
private ActivityMainDemoBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
/* 只要通过 DataBinging 加载布局 都会对应的生成 Binding对象,比如我们生成一个 MainActivityBinding对象
* https://blog.csdn.net/huangxin388/article/details/77425678
* 注意一定要把格式搞对了,才能自动生成 ActivityMainDemoBinding 对象
*/
binding = DataBindingUtil.setContentView(this, R.layout.activity_main_demo);
// 相当于 findViewById
// binding.mainTextview.setOnClickListener(new View.OnClickListener() {
// @Override
// public void onClick(View v) {
//
// }
// });
/**
* 在来试验 通过操作符给控件设置一些信息 (绑定基本数据类型和 String 类型)
* 直接给 Button 设置文本和 控件是否能被点击 ,需要在 xml 的配置文件配置一些 操作符
*/
// binding.setContent("对String类型数据的绑定");
// binding.setEnabled(false);//设置enabled值为false
// //给控件设置点击事件,发现其实点击无效,因为在布局文件中给cilckable属性绑定了enabled,在代码中设置enabled值为false,所以点击事件无效
// binding.mainTv2.setOnClickListener(new View.OnClickListener() {
// @Override
// public void onClick(View v) {
// Toast.makeText(DataBingingActivityDemo.this, "我被点击了", Toast.LENGTH_SHORT).show();
// }
// });
/**
* 在来试验通过 绑定 Model 数据类型
* binding.setVariable(BR.user,user);
*/
// DataModel data = new DataModel("绑定Model数据类型");
// binding.setVariable(BR.dataModel,data);
/**
* 在来试验:绑定事件,比如给控件设置 监听
* 1.android:onClick="@{event.click1}"
* 2.android:onClick="@{event::click2}"
* 3.android:onClick="@{()->event.cilck3(title4)}"
* [注]:()->event.cilck3(title4)是lambda表达式写法,
* 也可以写成:(view)->event.cilck3(title4),前面(view)表示onClick方法的传递的参数,
* 如果event.click3()方法中不需要用到view参数,可以将view省略。
*
* 当然event.click1也可以写成(view)->event.click1(view),将onClick(View view)的view参数传递给event.click1(view)方法。
* 大概就这意思,以下是伪代码
* onclick(View view){
* event.click1(view)
* }
*
*/
binding.setTitle1("事件绑定1");
binding.setTitle2("事件绑定2");
binding.setTitle3("事件绑定3");
binding.setTitle4("change ok"); // 对应 title4的值 作为cilck3(String s) 函数中的参数
binding.setEvent(new EventClickInterface() {
@Override
public void click1(View v) {
binding.setTitle1("事件1方法调用");
}
@Override
public void click2(View v) {
binding.setTitle2("事件2方法调用");
}
@Override
public void cilck3(String s) {
binding.setTitle3(s);
}
});
}
}
package com.application.databinging.event;
import android.view.View;
public interface EventClickInterface {
public void click1(View v);
public void click2(View v);
public void cilck3(String s);
}
总结 :上面就是 通过 DataBinding 的方式将传递代码逻辑给 抽象出来了:DataBinding基本原型就出来了
1: findViewById ------>bind.xxxx
2: view.setText() ----->在xml 中配置 下面属性
<data>
<variable name="title1" type="String" />
</data>
然后再 activity中:binding.settitle1 = "给控件设置属性"
3: view.setOnClickListener() ----->
<data>
<variable name="event"
type="com.application.databinging.event.EventClickInterface/>
</data>
binding.setEvent(new EventClickInterface() {
@Override
public void click1(View v) {
binding.setTitle1("事件1方法调用");
}
}
public interface EventClickInterface {
public void click1(View v);
}
推荐阅读