ContentResolver.notifyChange()不起作用

时间:2011-10-09 15:48:19

标签: android notifications android-contentprovider simplecursoradapter

一些背景资料:

当我运行我的应用程序时,前台活动会显示来自本地SQLite数据库的缓存信息,同时后台服务连接到服务器并获取一些新信息。检索数据时,服务会激活我的ContentProvider的insert方法,以更新我的本地SQLite数据库。插入完成后,我通知我的ContentObserver(前台活动)有关更改以刷新视图。

我知道插件工作正常,因为当我手动重新运行应用程序时,我可以在屏幕上看到新数据。不幸的是,自动刷新不起作用。调用方法ContentObserver.notifyChange()失败。最奇怪的是,它甚至没有停在catch (Exception e)线上。它直接在try/catch区块之外。我不知道是怎么回事。 LogCat,控制台很安静。

以下是我的ContentProvider中负责插入和更改通知的insert方法的代码部分。

SQLiteDatabase sqlDB = db.getWritableDatabase();
try {
    long newID = sqlDB.insertOrThrow(StatsCollectorDatabase.TABLE_USERS, null, values);
    if (newID > 0) {
        Uri newUri = ContentUris.withAppendedId(uri, newID);
        ContentResolver contentResolver = getContext().getContentResolver();
        contentResolver.notifyChange(uri, null); // This is the line that fails
        return newUri;
    } else {
        throw new SQLException("Failed to insert row into " + uri);
    }
} catch (SQLiteConstraintException e) {
    Log.i(DEBUG_TAG, "Ignoring constraint failure.");
} catch (Exception e) {
    Log.i(DEBUG_TAG, e.getLocalizedMessage());
}

以下是前台活动中onCreate方法的代码:

Cursor cursor = this.managedQuery(StatsCollectorProvider.CONTENT_URI, null, null, null, null);
String[] columns = new String[] { StatsCollectorDatabase._USER_NAME, StatsCollectorDatabase._USER_DEVICE_ID };
int[] views = new int[] { R.id.name_entry, R.id.number_entry };

SimpleCursorAdapter adapter = new SimpleCursorAdapter(this.getApplicationContext(), R.layout.list_item, cursor, columns, views, CursorAdapter.FLAG_REGISTER_CONTENT_OBSERVER);
this.setListAdapter(adapter);

Intent intent = new Intent(this.getApplicationContext(), UsersDownloaderService.class);
this.startService(intent);    

我尝试使用谷歌搜索,但没有发现任何相关内容。我该如何解决这个问题?

如果有必要,我可以提供更多代码。

2 个答案:

答案 0 :(得分:2)

我解决了。我的活动没有继承FragmentActivity,我的布局不包含任何片段,所以它无法工作 - 没有任何注册的ContentObservers。

现在看起来像这样:

首先,主要活动开始,注册片段布局并以onCreate方法启动后台服务:

super.onCreate(savedInstanceState);
setContentView(R.layout.list);

Intent intent = new Intent(this.getApplicationContext(), UsersDownloaderService.class);
this.startService(intent);  

这是我的list布局:

<?xml version="1.0" encoding="utf-8"?>
<fragment xmlns:android="http://schemas.android.com/apk/res/android"
    class="org.moyshe.cursors.ListViewFragment"
    android:name="org.moyshe.cursors.ListViewFragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >
</fragment>

班级ListViewFragment扩展ListFragment并实施LoaderCallbacks<Cursor>。这是我的ContentObserver,我一直在想念。如果有人对它的外观感兴趣,请阅读article - 这对我帮助很大。

我的服务和ContentProvider类没有改变。

答案 1 :(得分:1)

我遇到了类似的问题并找到了solution here

简而言之,事实证明我需要在我的内容提供商的setNotificationUri(ContentResolver cr, Uri uri)方法返回的游标上调用query()