1. 前言
2015年的Google IO大会上,Android 开发团队发布了一个数据绑定框架(Data Binding Library),官方原生支持 MVVM 模型。可以直接在 layout 布局 xml 文件中绑定数据,无需再 findViewById然后手工设置数据。
官方地址:https://developer.android.google.cn/topic/libraries/data-binding/index.html
2. Databinding框架的优势
Databinding是0反射的,所以它的性能比较好
可以直接在xml中绑定数据并实现一些处理逻辑,实时动态刷新数据。功能强大,可以节省很多手写的代码.
有效减少绑定应用程序逻辑和布局所需的一些耦合性代码 ,简化代码
3. Databinding原理
针对每个Activity的布局,在编译阶段,生成一个ViewDataBinding类的对象,该对象持有Activity要展示的数据和布局中的各个view的引用,解决了令人厌烦的findViewById问题,同时该对象还有如下功能:
* 将数据分解到各个view
* 在UI线程上更新数据
* 监控数据的变化,实时更新
有了这些功能,我们要展示的数据已经和展示它的布局紧紧绑定在了一起。
4. Databinding的引入
- DataBinding需要Android Gradle插件的支持,版本至少在1.5以上,需要的Android studio的版本在1.3以上。
- 在Android Studio上使用,需要在module级别的build.gradle上添加对DataBinding的支持:
- android {
dataBinding {
enabled = true
}
}
4. 避免反射
DataBinding使用的是代码生成技术。在Build的时候DataBinding会生成一些额外的类。
这些类在build-generated-source-apt-debug-your-package-name-databinding目录下可以看到
这些类也会随之打包到apk中,因此它拥有比较好的性能。
5. Databinding基本使用
<?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">
<data>
<variable
name="user"
type="com.example.databinding.databindingdemo.model.UserInfo" />
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:text="@{user.nickname ?? user.name}" />
</LinearLayout>
</layout>
注:@{user.nickname ?? user.name}相当于三元运算符 TextUtile.isEmpty(user.nickname) ? user.name : user.nickname
package com.example.databinding.databindingdemo;
import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import com.example.databinding.databindingdemo.adapter.AppAdapter;
import com.example.databinding.databindingdemo.databinding.ActivityMainBinding;
import com.example.databinding.databindingdemo.handler.MyClick;
import com.example.databinding.databindingdemo.listener.UserClickListener;
import com.example.databinding.databindingdemo.model.UserInfo;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity implements UserClickListener {
private ActivityMainBinding mainBinding;
private UserInfo userInfo;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mainBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
userInfo = new UserInfo();
userInfo.setName("name1");
mainBinding.setUser(userInfo);
}
}
package com.example.databinding.databindingdemo.model;
import android.databinding.BaseObservable;
import android.databinding.Bindable;
import android.databinding.BindingAdapter;
import android.widget.ImageView;
import com.example.databinding.databindingdemo.BR;
import com.example.databinding.databindingdemo.R;
/**
* Created by user on 2018/3/8.
*/
public class UserInfo extends BaseObservable{
private String name;
@Bindable
private String nickname;
@Bindable
private int resId = R.mipmap.ic_launcher;
private int paLeft = 100;
@Bindable
private boolean vip = true;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getNickname() {
return nickname;
}
public void setNickname(String nickname) {
this.nickname = nickname;
notifyPropertyChanged(BR.nickname);
}
public int getResId() {
return resId;
}
public void setResId(int resId) {
this.resId = resId;
notifyPropertyChanged(BR.resId);
}
public int getPaLeft() {
return paLeft;
}
public void setPaLeft(int paLeft) {
this.paLeft = paLeft;
}
public boolean isVip() {
return vip;
}
public void setVip(boolean vip) {
this.vip = vip;
}
@BindingAdapter("imageUrl")
public static void loadImage(ImageView imageView, int resId){
imageView.setImageResource(resId);
}
}