Observable觀察者
我們知道绎巨,Data Binding中如果我們直接修改Model實(shí)體對(duì)象(也就是POJO)中的數(shù)據(jù)拣挪,這些數(shù)據(jù)并不能直接更新到UI涌哲,所以Data Binding給了我們一套很好的通知機(jī)制肤寝,分別有三類: Observable objects, observable fields, and observable collections育苟,分別表示觀察對(duì)象筹裕、觀察字段醋闭、觀察集合,若相應(yīng)的對(duì)象饶碘、字段目尖、集合中數(shù)據(jù)變化時(shí)候,那么UI將會(huì)自動(dòng)更新數(shù)據(jù)扎运。下面我們一一來介紹它們的用法:
Observable objects
因?yàn)镺bservable是個(gè)接口瑟曲,Google為我們提供了一個(gè)BaseObservable類,我們只要把Model類繼承自它就獲得了通知UI更新數(shù)據(jù)的能力了豪治,再getter方法上添加Bindable注解洞拨,在setter方法中使用notifying提醒UI更新數(shù)據(jù)。
private static class User extends BaseObservable {
private String userName;
private String userPassword;
public User(String userName, String userPassword) {
this.userName = userName;
this.userPassword = userPassword;
}
@Bindable
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
notifyPropertyChanged(BR.userName);
}
@Bindable
public String getUserPassword() {
return userPassword;
}
public void setUserPassword(String userPassword) {
this.userPassword = userPassword;
notifyPropertyChanged(BR.userPassword);
}
}
首先我們需要在getter方法上添加Bindable注解后负拟,Bindable注解會(huì)自動(dòng)生成一個(gè)BR類烦衣,該類位于app module包下,通過BR類我們?cè)O(shè)置更新的數(shù)據(jù)掩浙,當(dāng)Model中的數(shù)據(jù)發(fā)生變化時(shí)花吟,setter方法中的notifyPropertyChanged()就會(huì)通知UI更新數(shù)據(jù)了。
View(Activity)展示
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
NamePasswordBinding namePasswordBinding= DataBindingUtil.setContentView(this,R.layout.name_password);
final com.bean.User user=new User("Michael","1234");
namePasswordBinding.setUser(user);
namePasswordBinding.loginBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
user.setUserName("qianrushi");
user.setUserPassword("123456");
}
});
}
效果如下:
ObservableFields
我們剛剛介紹的通知UI更新的方法是用User類繼承自BaseObservable厨姚,然后在getter上添加注解衅澈、在setter中添加notify方法,這感覺總是有點(diǎn)麻煩谬墙,步驟繁瑣今布,于是经备,Google推出ObservableFields類斧蜕,使用它我們可以簡化我們的Model類镜豹,如:
public class User{
public final ObservableField<String> userName = new ObservableField<>();
public final ObservableField<String> userPassword = new ObservableField<>();
}
onCreate()方法中的代碼就變成了這樣:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
NamePasswordBinding namePasswordBinding= DataBindingUtil.setContentView(this,R.layout.name_password);
final com.bean.User user=new User();
user.userName.set("Michael");
user.userPassword.set("1234");
namePasswordBinding.setUser(user);
namePasswordBinding.loginBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
user.userName.set("qianrushi");
user.userPassword.set("123456");
}
});
}
效果如下:
當(dāng)然ObservableField<T>中傳入的泛型可以是Java中的基本類型,當(dāng)然我們還可以使用 ObservableBoolean, ObservableByte, ObservableChar, ObservableShort, ObservableInt, ObservableLong, ObservableFloat, ObservableDouble, ObservableParcelable等具體的類型第队,效果也和ObservableField<T>是一樣的傅蹂,如:
public class User{
public final ObservableField<String> userName = new ObservableField<>();
public final ObservableField<Integer> userPassword = new ObservableField<>();
public final ObservableInt userAge = new ObservableInt();
}
Observable Collections
Google也為我們提供了一些通知類型的集合纷闺,有這三種:ObservableArrayList<T>、ObservableArrayMap<K,V>贬派、ObservableMap<K,V>急但,它和平場(chǎng)使用的List、Map用法一樣搞乏,但是多了通知功能波桩。
我們?cè)趌ayout中的<data>區(qū)域?qū)氚缶涂梢灾苯佑盟耍?dāng)它內(nèi)部的數(shù)據(jù)發(fā)生改變時(shí)就自動(dòng)會(huì)通知UI界面更新请敦。
下面xml布局使用ObservableMap<K,V>镐躲,當(dāng)map中的數(shù)據(jù)改變時(shí)候同時(shí)也通知了UI界面更新。
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<data>
<import type="android.databinding.ObservableMap"/>
<import type="com.sunzxyong.binding.Keys"/>
<variable
name="map"
type="ObservableMap<String,Object>"/>
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{map[Keys.name]}" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{String.valueOf(1+(Integer)map[Keys.age])}" />
<Button
android:id="@+id/btn"
android:layout_marginTop="30dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="changeData" />
</LinearLayout>
</layout>
onCreate()方法:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding mBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
final ObservableMap<String, Object> map = new ObservableArrayMap<>();
map.put("name", "sunzxyong");
map.put("age", 22);
mBinding.setMap(map);
mBinding.btn.setOnClickListener(new android.view.View.OnClickListener() {
@Override
public void onClick(android.view.View v) {
map.put("name","hello");
map.put("age",20);
}
});