在android 开发过程中,也许有人会经常遇到这样的问题:CursorAdapter与一个list页面绑定,所有的代码都正确,CursorAdapter 与 Cursor也正常关联,但当数据库改变时list页面就是不更新!下面我们来看看源代码:
CursorAdapter.java
void init(Context context, Cursor c, boolean autoRequery) {
// ChangeObserver 继承 ContentObserver
mChangeObserver = new ChangeObserver();
//交给Cursor注册Observer
c.registerContentObserver(mChangeObserver);
}
从上面的代码可以看出,当CursorAdapter与Cursor关联时,向Cursor注册了一个ContentObserver。看一下Cursor 的registerContentObserver:
void registerContentObserver(ContentObserver observer) {
//这里mContentObservable是ContentObservable的一个实例
mContentObservable.registerObserver(observer);
}
public class ContentObservable extends Observable<ContentObserver> {
public void registerObserver(ContentObserver observer) {}
public void dispatchChange(boolean selfChange) {}
public void notifyChange(boolean selfChange) {}
}
public abstract class Observable<T> {
protected final ArrayList<T> mObservers = new ArrayList<T>();
public void registerObserver(T observer) {}
public void unregisterObserver(T observer) {}
public void unregisterAll() {
}
其实Cursor中仅仅只用ArrayList<T> mObservers 这个链表保存了注册进来的ContentObserver,然后啥也没有做!那我们自然要问这些个observer是什么时候被触发的呢 ?继续看看Cursor.java 的源码:
Cursor.java
public void setNotificationUri(ContentResolver cr, Uri notifyUri) {
mNotifyUri = notifyUri;
mContentResolver = cr;
mSelfObserver = new SelfContentObserver(this);
//进行Observer的注册
mContentResolver.registerContentObserver(
mNotifyUri, true, mSelfObserver);
}
protected void onChange(boolean selfChange) {
//触发所有的Observer
mContentObservable.dispatchChange(selfChange);
}
//ContentObservable.java
public void dispatchChange(boolean selfChange) {
for (ContentObserver observer : mObservers) {
if (!selfChange || observer.deliverSelfNotifications()) {
observer.dispatchChange(selfChange);
}
}
}
现在我们清楚了,原来Cursor在函数setNotificationUri()中向ContentResolver注册了一个mSelfObserver,当Cursor的mSelfObserver被触发时它会回调自己链表中的所有Observer,将事件传递给所有监听者(也就是上面提到的CursorAdapter)。
然而,setNotificationUri()是在什么地方被调用的呢?这就是本文要说的重点。但凡有关CursorAdapter页面不更新的问题,基本都是这个函数没有被调用造成的。setNotificationUri()这个需要我们在实现ContentProvider的query函数时,手动调用,系统是不会自动帮我们调用的:
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
QLiteDatabase db = getWritableDatabase();
Cursor result = qb.query(db, projection, args.where, args.args, null, null, sortOrder);
result.setNotificationUri(getContext().getContentResolver(), uri);
return result;
}
切记,一定要调用setNotificationUri(),这样返回的Cursor与CursorAdapter关联时才能够自动刷新页面!
另外在实现 ContentProvider的 update, delete, insert, bulkInsert 这些函数时,在函数末尾千万别忘了调用notifyChange(),只有这样通过ContentResolver注册的Observer才会被触发:
getContext().getContentResolver().notifyChange(uri, null);