天天看點

Android官方DataBinding(二):動态資料更新notifyPropertyChanged

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 

繼續閱讀