Android官方DataBinding(二):動态資料更新notifyPropertyChanged
附錄文章1使用了最簡單的Android官方DataBinding技術,是以寫的例子以簡單明了能說明問題即可。但是附錄文章1的資料不能更新,即如果當使用者的資料User的内部變量發生變化時候,不能反映到View層,即TextView不能發生變化,如果是這樣,代碼是沒有意義的,因為現實的開發場景,User的資料肯定要動态變化的,并且動态變化的資料model肯定要反映在view上。本例就繼續使用Android官方DataBinding技術,實作動态資料變化更新到View。
本例要實作一個目的就是,假設附錄文章1的User資料模型變化,則變化内容實時更新同步到綁定的view上。
改造附錄文章1的部分内容實作上述目标。
(1)在User的資料模型定義上:
package zhangphil.test;
import android.databinding.BaseObservable;
import android.databinding.Bindable;
/**
* Created by Phil on 2017/8/17.
*/
public class User extends BaseObservable {
private String id;
private String name;
private String blog;
public User() {
}
public User(String id, String name, String blog) {
this.id = id;
this.name = name;
this.blog = blog;
}
public void setId(String id) {
this.id = id;
notifyPropertyChanged(BR.id);
}
@Bindable
public String getId() {
return this.id;
}
public void setName(String name) {
this.name = name;
notifyPropertyChanged(BR.name);
}
@Bindable
public String getName() {
return this.name;
}
public void setBlog(String blog) {
this.blog = blog;
notifyPropertyChanged(BR.blog);
}
@Bindable
public String getBlog() {
return this.blog;
}
}
其實變化比較小,在get方法上增加注解@Bindable,在User的set方法内用notifyPropertyChanged通知資料變化,資料模型變化後由Android系統通知get然後重新整理view。
(2)和附錄文章1相比,在xml布局裡面為TextView增加id:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable
name="user"
type="zhangphil.test.User" />
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:id="@+id/id"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.id}"
android:textColor="@android:color/holo_red_light" />
<TextView
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.name}"
android:textColor="@android:color/holo_red_light" />
<TextView
android:id="@+id/blog"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.blog}"
android:textColor="@android:color/holo_red_light" />
</LinearLayout>
</layout>
這裡增加id,是為了後面通過ActivityMainBinding拿到具體的TextView對象,切換到原始的TextView對象操作。
(3)上層Java代碼寫一個簡單的示範例子,在一個線程内每隔2秒更新一次User的資料字段,當id值自增長到5時候,切換到原始TextView的setText方法實作設定:
package zhangphil.test;
import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
import zhangphil.test.databinding.ActivityMainBinding;
public class MainActivity extends AppCompatActivity {
private String name = "zhangphil";
private int id = 1;
private String blog = "http://blog.csdn.net/zhangphil";
private User user = new User();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
binding.setUser(user);
final TextView idView = binding.id;
final TextView nameView = binding.name;
final TextView blogView = binding.blog;
new Thread(new Runnable() {
@Override
public void run() {
while (true) {
runOnUiThread(new Runnable() {
@Override
public void run() {
if (id == 5) {
idView.setText("zhang");
nameView.setText("phil");
blogView.setText("blog");
} else {
user.setId(String.valueOf(id++) + " " + System.currentTimeMillis());
user.setName(name + " " + System.currentTimeMillis());
user.setBlog(blog + " " + System.currentTimeMillis());
}
}
});
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
}
}
本例之是以示範id==5時候,回退到原始的TextView進行操作,是考慮到在某些必要時候,需要拿到View進行顯示/隐藏操作。
代碼運作結果:
當線程運作到id=5時候:
附錄:
1,《Android官方DataBinding簡例(一)》連結:http://blog.csdn.net/zhangphil/article/details/77322530