24

我一直在开发一个小型的待办事项列表应用程序。我使用 CursorLoader 从内容提供者更新 ToDolistview。我写了一个函数onNewItemAdded(),当用户在文本视图中输入一个新项目并单击 Enter 时调用该函数。参考以下:

public void onNewItemAdded(String newItem) {
    ContentResolver cr = getContentResolver();

    ContentValues values = new ContentValues();
    values.put(ToDoContentProvider.KEY_TASK, newItem);

    cr.insert(ToDoContentProvider.CONTENT_URI, values);
    // getLoaderManager().restartLoader(0, null, this); // commented for the sake of testing
}

@Override
protected void onResume() {
    super.onResume();
    //getLoaderManager().restartLoader(0, null, this); // commented for the sake of testing
}

public Loader<Cursor> onCreateLoader(int id, Bundle args) {

    CursorLoader loader = new CursorLoader(this,
            ToDoContentProvider.CONTENT_URI, null, null, null, null);
    Log.e("GOPAL", "In the onCreateLoader");
    return loader;
}

public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {

    int keyTaskIndex = cursor.getColumnIndexOrThrow(ToDoContentProvider.KEY_TASK);
    Log.e("GOPAL", "In the onLoadFinished");
    todoItems.clear();
    if (cursor.moveToNext() == false) Log.e("GOPAL", "Empty Cursor");
    else {
        while (cursor.moveToNext()) {
            ToDoItem newItem = new ToDoItem(cursor.getString(keyTaskIndex));
            todoItems.add(newItem);
        }
        aa.notifyDataSetChanged(); // aa is arrayadapter used for the listview
    }
}

我已阅读,只要内容提供程序数据库中的数据发生更改,CursorLoader 就会自动更新视图。这意味着我想,getLoaderManager().restartLoader(0, null, this)只要数据发生变化,就必须隐式调用,对吗?但这并没有发生。每当我添加一个新项目(该项目从 db 添加到数据库onNewItemAdded,但没有显式调用 restartLoader )时,暂停此活动并恢复它。我没有看到对 restartLoader 的任何隐式调用(即使 db 已更改),并且列表视图也没有更新添加的新项目。这是为什么?CursorLoader即使应用程序未处于活动状态,如何自动更新视图???谢谢 :)

编辑:我也用于getContext().getContentResolver().notifyChange(insertedId, null)插入我的内容提供者。

4

3 回答 3

72

我找到了我的问题的答案。通常,CursorLoader 不会自动检测数据更改并将其加载到视图中。我们需要跟踪 URI 的变化。这可以通过以下步骤来完成:

  1. 使用游标在内容解析器中注册观察者:(在 ContentProvider 的查询方法中完成)
    cursor.setNotificationUri(getContext().getContentResolver(), uri);

  2. insert()现在,当使用// URI 底层数据发生任何变化时,delete()我们使用以下update(),命令通知ContentResolver更改:

    getContext().getContentResolver().notifyChange(insertedId, null);

  3. 这是由观察者接收到的,我们在步骤 1 中注册并调用了ContentResolver.query(),这又调用了ContentProviderquery()方法以将新的光标返回到LoaderManagerLoaderManager调用onLoadFinished()传递此游标,以及我们用新数据CursorLoader更新视图(使用)的位置。Adapter.swapCursor()

对于自定义 AsyncTaskLoaders:

有时我们需要我们的自定义加载器而不是 CursorLoader。在这里,我们可以使用光标以外的其他对象来指向加载的数据(如列表等)。在这种情况下,我们将没有特权通过光标通知 ContentResolver。应用程序也可能没有内容提供程序来跟踪 URI 更改。在这种情况下,我们使用 BroadcastReceiver 或显式 ContentObserver 来实现自动视图更新。如下:

  1. 我们需要定义我们的自定义加载器,它扩展AsyncTaskLoader并实现了它的所有抽象方法。与 不同CursorLoader的是,我们的自定义加载器可能会或可能不会使用内容提供程序,并且它的构造函数可能不会调用ContentResolver.query(),当这个加载器被实例化时。所以我们使用广播接收器来达到目的。
  2. 我们需要实例化一个BroadCastReceiver或抽象类 ContentObserver的方法。OnStartLoading()AsyncTaskLoader
  3. 这个广播接收器应该被定义为从内容提供者或任何系统事件(如新安装的应用程序)接收数据更改广播,并且它必须调用加载程序的onContentChanged()方法,以通知加载程序有关数据更改。Loader 自动完成其余工作以加载更新的数据并调用onLoadFinished()以更新视图。

有关更多详细信息,请参阅:http: //developer.android.com/reference/android/content/AsyncTaskLoader.html

我发现这对于清晰的解释非常有用:http ://www.androiddesignpatterns.com/2012/08/implementing-loaders.html

于 2013-03-20T16:54:44.360 回答
3

好吧,我认为您可以在某些事件上重新启动加载程序。例如,就我而言,我有一项 TODO 活动。单击“添加”选项时,它会启动新的活动,该活动具有提供新 TODO 的视图。

我在父活动的 onActivityResult() 中使用以下代码

getLoaderManager().restartLoader(0, null, this);

这对我来说可以。如果有更好的方法,请分享。

于 2014-06-20T13:48:03.613 回答
2

在初始化时获取对加载器的引用,如下所示

 Loader dayWeatherLoader = getLoaderManager().initLoader(LOADER_DAY_WEATHER, null, this);

然后创建一个扩展 ContentObserver 的类,如下所示

 class DataObserver extends ContentObserver {

    public DataObserver(Handler handler) {
        super(handler);
    }

    @Override
    public void onChange(boolean selfChange, Uri uri) {
        dayWeatherLoader.forceLoad();
    }
}

然后在 onResume 生命周期方法中注册内容观察者如下

@Override
public void onResume() {
    super.onResume();
    getContext().getContentResolver().registerContentObserver(CONTENTPROVIDERURI,true,new DayWeatherDataObserver(new Handler()));
}

每当 content provider 底层数据发生变化时,就会调用 contentobserver 的 onChange 方法,可以让 loader 重新加载数据

于 2017-03-09T01:23:34.893 回答