依赖
// module 的 build.gradle 中的 android 节点增加此代码
android {
dataBinding {
enabled = true
}
}
// gradle.properties 中增加此行
android.databinding.enableV2=true
使用
// 1、对于布局文件 alt + enter,选择转换成 databing 布局
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!" />
</LinearLayout>
</layout>
// 2、新增一个Bean
package com.example.xubinhong.xbh_android_test.bean;
public class Student {
public String name;
}
// 3、XML 中进行绑定,然后同步下项目
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable
name="student"
type="com.example.xubinhong.xbh_android_test.bean.Student"/>
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{student.name,default = Hong}" />
</LinearLayout>
</layout>
// 4、app/build/generated/data_binding_base_class_source_out... 下自动生成了文件
public abstract class ActivityMainBinding extends ViewDataBinding {
@NonNull
public final TextView tv;
@Bindable
protected Student mStudent;
protected ActivityMainBinding(DataBindingComponent _bindingComponent, View _root,
int _localFieldCount, TextView tv) {
super(_bindingComponent, _root, _localFieldCount);
this.tv = tv;
}
public abstract void setStudent(@Nullable Student student);
@Nullable
public Student getStudent() {
return mStudent;
}
@NonNull
public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater,
@Nullable ViewGroup root, boolean attachToRoot) {
return inflate(inflater, root, attachToRoot, DataBindingUtil.getDefaultComponent());
}
@NonNull
public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater,
@Nullable ViewGroup root, boolean attachToRoot, @Nullable DataBindingComponent component) {
return DataBindingUtil.<ActivityMainBinding>inflate(inflater, com.example.xubinhong.xbh_android_test.R.layout.activity_main, root, attachToRoot, component);
}
@NonNull
public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater) {
return inflate(inflater, DataBindingUtil.getDefaultComponent());
}
@NonNull
public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater,
@Nullable DataBindingComponent component) {
return DataBindingUtil.<ActivityMainBinding>inflate(inflater, com.example.xubinhong.xbh_android_test.R.layout.activity_main, null, false, component);
}
public static ActivityMainBinding bind(@NonNull View view) {
return bind(view, DataBindingUtil.getDefaultComponent());
}
public static ActivityMainBinding bind(@NonNull View view,
@Nullable DataBindingComponent component) {
return (ActivityMainBinding)bind(component, view, com.example.xubinhong.xbh_android_test.R.layout.activity_main);
}
}
// 5、使用
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
Student student = new Student();
student.name = "first";
binding.setStudent(student);
}
}
此后只需要修改数据,然后调用对应的set方法,即可更新UI
原理分析
AS编译时扫描layout文件,并解析layout文件,生成一个binding类
比如上面的数据Student
上面的View,一个根布局LinearLayout,一个TextView
这3者都会被保存在这个文件的成员变量中
DataBindingUtil.setContentView会调用activiyt.setContentView
通过其XmlParser解析出根View
再给对应View字段赋值
调用binding.setStudent,会对Student字段赋值
并且刷新UI
对于布局文件中描述的对应规则,进行赋值
为何使用它?
用在MVVM中较多,有兴趣可以看看MVC、MVP、MVVM异同一文:
https://www.jianshu.com/p/81878f0ba5e9
后记
有什么写得错误、让人费解或遗漏的地方,希望可以不吝赐教,我会马上更改