如何使用ContentProvider在进行许多更改时暂停向观察者发送通知

时间:2010-11-27 14:46:12

标签: android cursor observer-pattern android-contentprovider suspend

我有一个ExpandableListView,它使用SimpleCursorTreeAdapter,它使用ContentProvider返回的游标。这很好,因为它始终与数据保持同步,但有时我需要对数据库进行许多更改,以便在同一秒内多次重新获取光标。是否可以暂停ContentObservers的通知以避免不必要的requerys?

2 个答案:

答案 0 :(得分:1)

可能的解决方案是修改内容提供商以允许暂停通知。要通知的URI将添加到队列中,直到禁用暂停。

private boolean suspendNotifications = false;
private LinkedList<Uri> suspendedNotifications = new LinkedList<Uri>();
private HashSet<Uri> suspendedNotificationsSet = new HashSet<Uri>();

    private void notifyChange(Uri uri) {
    if (suspendNotifications) {
        synchronized (suspendedNotificationsSet) { // Must be thread-safe
            if (suspendedNotificationsSet.contains(uri)) {
                // In case the URI is in the queue already, move it to the end.
                // This could lead to side effects because the order is changed
                // but we also reduce the number of outstanding notifications.
                suspendedNotifications.remove(uri); 
            }
            suspendedNotifications.add(uri);
            suspendedNotificationsSet.add(uri);
        }
    }
    else {
        getContext().getContentResolver().notifyChange(uri, null);
    }
}

private void notifyOutstandingChanges() {
    Uri uri;
    while ((uri = suspendedNotifications.poll()) != null) {
        getContext().getContentResolver().notifyChange(uri, null);
        suspendedNotificationsSet.remove(uri);
    }
}

private void setNotificationsSuspended(boolean suspended) {
    this.suspendNotifications = suspended;
    if (!suspended) notifyOutstandingChanges();
}

@Override
public Uri insert(Uri uri, ContentValues values) {
    ...
    notifyChange(uri);
    return newItemUri;
}

我不确定如何最好地启用/禁用暂停,但有一种可能性是在update()方法中有一个特殊的URI来打开/关闭暂停(例如content://&lt; authority&gt; / suspension) :

@Override
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
    switch (uriMatcher.match(uri)) {
    ...
    case SUSPEND:
        boolean enabled = values.getAsBoolean("enabled");
        setNotificationsSuspended(enabled);
        break;
    ... 
    }
}

对数据库进行更改的服务现在可以在启动时挂起ContentProvider,并在完成时禁用暂停。

答案 1 :(得分:0)

您可以使用http://developer.android.com/reference/android/widget/BaseAdapter.html#unregisterDataSetObserver(android.database.DataSetObserver)取消注册您的听众吗?一旦您的工作准备就绪,请再次注册?对我来说听起来像AsyncTask。