MVC,MVP都是大家所熟知的,而今天的主题的MVVM数据绑定,这个是最新推出的框架,刚加入公司没多久,看公司里的同事再用这个,所以我熟悉之后发现确实挺好用的,在这里分享给大家。
Data Binding自从去年的Google I/O发布到至今,也有近一年的时间了。这一年来,从Beta到如今比较完善的版本,从Android Studio 1.3到如今2.1.2的支持,可以说Data Binding已经是一个可用度较高,也能带来实际生产力提升的技术了。
而事实上,真正使用到Data Binding的公司、项目仍然是比较少的。可能是出于稳定性考虑,亦或是对Data Binding技术本身不够熟悉,又或许对新技术没什么追求。
我司在新的产品中就全面使用了Data Binding技术,无论是我,还是新来直接面对Data Binding上手的工程师也好,都对其爱不释手,用惯了后简直停不下来。
希望在看完本文的介绍后,会有更多的朋友产生兴趣,来使用Data Binding,参与它的讨论。
这里我以我的理解来说,就是一个或者多个实体对象对应着一个页面,实体中的某些属性绑定着页面上的控件,当属性的值改变时,页面上的控件会自动更新数据。
XML布局
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context="com.example.junwen.databinding.MainActivity"> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="姓名:" /> <TextView android:id="@+id/activity_main_stu_name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="学生姓名" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="年龄:" /> <TextView android:id="@+id/activity_main_stu_age" android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="学生年龄" /> </LinearLayout> <Button android:id="@+id/activity_main_stu_setting" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="更改姓名" /> </LinearLayout>Student实体类
/** * 描述:学生对象 * 作者:卜俊文 * 创建:2016/8/10 10:20 * 邮箱:344176791@qq.com */ public class Student { private String name; private int age; public Student(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } }MainActivity主页面
public class MainActivity extends AppCompatActivity { private TextView tv_name; //学生姓名 private TextView tv_age; //学生年龄 private Button btn_setting; //设置 private Student student; //学生对象 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); initData(); initListener(); } /** * 描述:初始化控件 * 作者:卜俊文 * 邮箱:344176791@qq.com * 创建时间: 2016/8/10 10:17 */ private void initView() { tv_name = (TextView) findViewById(R.id.activity_main_stu_name); tv_age = (TextView) findViewById(R.id.activity_main_stu_age); btn_setting = (Button) findViewById(R.id.activity_main_stu_setting); } /** * 描述:初始化数据 * 作者:卜俊文 * 邮箱:344176791@qq.com * 创建时间: 2016/8/10 10:19 */ private void initData() { //创建一个学生对象 student = new Student("俊文", 22); //根据学生对象赋值到控件上 tv_name.setText(student.getName()); tv_age.setText(String.valueOf(student.getAge())); } /** * 描述:初始化监听 * 作者:卜俊文 * 邮箱:344176791@qq.com * 创建时间: 2016/8/10 10:22 */ private void initListener() { btn_setting.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { //点击设置,更改Student对象的名字。 student.setName("卜俊文"); tv_name.setText(student.getName()); } }); } }
XML布局
<layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto"> <data> <variable name="student" type="com.example.junwen.databinding.Student" /> </data> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="姓名:" /> <TextView android:id="@+id/activity_main_stu_name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="@{student.name}" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="年龄:" /> <TextView android:id="@+id/activity_main_stu_age" android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dp" android:text="@{String.valueOf(student.age)}" /> </LinearLayout> <Button android:id="@+id/activity_main_stu_setting" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="更改姓名" /> </LinearLayout> </layout>Student实体类
/** * 描述:学生对象, @Bindable注解是为了能在BR里面找到这个属性 notifyPropertyChanged(); 这个方法是当调用的时候,会通知绑定的控件去改变值 * 作者:卜俊文 * 创建:2016/8/10 10:20 * 邮箱:344176791@qq.com */ public class Student extends BaseObservable { @Bindable private String name; @Bindable private int age; public Student(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; notifyPropertyChanged(com.example.junwen.databinding.BR.name); } public int getAge() { return age; } public void setAge(int age) { this.age = age; notifyPropertyChanged(com.example.junwen.databinding.BR.age); } }MainActivity主页面
public class MainActivity extends AppCompatActivity { private ActivityMainBinding activity_main_binding; //本页面的Binding对象 private Student student; //学生对象 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initData(); initListener(); } /** * 描述:初始化数据 * 作者:卜俊文 * 邮箱:344176791@qq.com * 创建时间: 2016/8/10 10:48 */ private void initData() { //这个就类似于setContentView activity_main_binding = DataBindingUtil.setContentView(this, R.layout.activity_main); student = new Student("俊文", 22); //把Student对象绑定到布局的控件中去,现在开始只要student对象属性变化,控件的值也是相应改变 activity_main_binding.setStudent(student); } /** * 描述:初始化监听 * 作者:卜俊文 * 邮箱:344176791@qq.com * 创建时间: 2016/8/10 10:22 */ private void initListener() { activity_main_binding.activityMainStuSetting.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { student.setName("卜俊文"); } }); } }
(1) DataBinding不需要findviewbyId,只需要根据binding对象去查找到对应的控件。
(2)直接绑定一个对象到XML布局中,当对象的属性变化时,布局中的控件会马上同步变化。
(3)UI代码放到了xml中,布局和数据更紧密
(1)IDE支持还不那么完善(提示、表达式)
(2)报错信息不那么直接
dataBinding { enabled = true }
<layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto"> <data> <variable name="item" type="com.example.junwen.databinding.Student" /> </data> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@{item.name}" /> </LinearLayout> </layout>
此模块由<Layout>标签作为根节点,里面包含两部分,第一部分是<data>,第二部分就是<LinearLayout>,也就类似于以前我们写布局时的根布局。
其中<data>标签中可以声明对象,变量以及导入包等功能,在这里我导入了Student对象,在下面我需要用到我就可以导入进来,并且起成item的一个别名。
其中<LinearLayout>中,包含了一个TextView,这个TextView的text属性值是这样赋值的(格式必须对):
android:text="@{item.name}"双引号中 " @{具体变量或者对象的属性值}" ,不只这些,还可以在里面运用表达式,例如上面的例子中的
android:text="@{String.valueOf(student.age)}"(1)取得Binding对象,这个就类似于setContentView(),返回的对象其实是一个ViewDataBinding,但是我这里写的是ActivityMainBinding,你可以比较一下ActivityMainBinding 和 R.layout.activity_main有什么相似的地方吗?不错,就是拼接起来的,这样就可以获得这个页面的Binding对象。
ActivityMainBinding activity_main_binding = DataBindingUtil.setContentView(this, R.layout.activity_main);(2)创建一个Student对象,并且调用setStudent(student)方法,这个方法他会自动生成的,实现了实体与xml绑定。
Student student = new Student("俊文", 22); //把Student对象绑定到布局的控件中去,现在开始只要student对象属性变化,控件的值也是相应改变 activity_main_binding.setStudent(student)(3)设置监听,直接通过binding对象查询到按钮控件,这个activityMainStuSetting也会自动生成的,如果遇到没有生成,你需要build一下项目,他就会出来了,执行setName方法,在其内部就会去通知控件更新文本。
activity_main_binding.activityMainStuSetting.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { student.setName("卜俊文"); } });
1229.gif
Android官方数据绑定框架DataBinding 从零开始的Android新项目7 - Data Binding入门篇 Android官方数据绑定框架DataBinding(一) Android Data Binding代码实战 完全掌握Android Data Binding 数据绑定DataBinding
在此就总结了一些基本的DataBind的用法,可能有些地方说的不到位的,也是刚接触不是很熟悉,就是凭自己的理解来说的。
