【问题标题】:Android ListFragment data not updating via SimpleCursorAdapter / LoaderManager / notifyChange()Android ListFragment 数据未通过 SimpleCursorAdapter / LoaderManager / notifyChange() 更新
【发布时间】:2014-10-28 13:03:01
【问题描述】:

我有一个 ListFragment,它显示了存储在我的数据库表中的所有购物清单的名称。

问题是当我向表中添加新的购物清单行时,UI 上的 ListFragment 不会自动更新。 (只有关闭并重新启动应用程序才能看到更改。)

首先,这是我添加购物清单时在DbContentProvider 类中执行的代码:

`

            // Insert the values into the table
            id = db.insert(SHOPPING_LISTS_META_TABLE, null, values);

            if (id > -1) {
                // Construct and return the URI of the newly inserted row.
                Uri insertedId = ContentUris.withAppendedId(CONTENT_URI_SHOPPING_LISTS_META, id);

                // Notify any observers of the change in the data set.
                Log.d(LOG_TAG, "................. notifyChange(\"" + insertedId + "\", null)");
                getContext().getContentResolver().notifyChange(insertedId, null);
                Log.d(LOG_TAG, "................. notifyChange() done");
                return insertedId;
            }
            else {
                return null;
            }

`

...这里是它的 LogCat 输出...

10-28 12:29:41.133: D/SQLiteOpenHelper(19401): ................. notifyChange("content://org.example.myapp.DbContentProvider/shopping_lists_meta/12", null) 10-28 12:29:41.143: D/SQLiteOpenHelper(19401): ................. notifyChange() done 10-28 12:29:41.153: D/HomeActivity(19401): Shopping list, "My Test Shopping List" created: content://org.example.myapp.DbContentProvider/shopping_lists_meta/12 10-28 12:29:41.183: D/AbsListView(19401): unregisterIRListener() is called 10-28 12:29:41.193: E/ViewRootImpl(19401): sendUserActionEvent() mView == null 10-28 12:29:41.503: D/AbsListView(19401): unregisterIRListener() is called

在 LogCat 中,当我添加新的购物清单行时,我的 ListFragment 类根本没有输出。这是我的 ListFragment 类...

`

公共类 ShoppingListNamesListFragment 扩展 ListFragment 实现 LoaderManager.LoaderCallbacks {

private final static String LOG_TAG = ShoppingListNamesListFragment.class.getSimpleName(); 

// This is the Adapter being used to display the list's data
public static SimpleCursorAdapter mAdapter;

// These are the Contacts rows that we will retrieve
static final String[] PROJECTION = {DbContentProvider.KEY_ID,
    DbContentProvider.KEY_SHOPPING_LIST_NAME,
    DbContentProvider.KEY_IS_SHOPPING_LIST_SELECTED};

// This is the select criteria
static final String SELECTION = null;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    Log.d(LOG_TAG, "................. onCreate()");

}

@Override
public void onActivityCreated(Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);

    Log.d(LOG_TAG, "................. onActivityCreated()");

    // For the cursor adapter, specify which columns go into which views
    String[] fromColumns = {DbContentProvider.KEY_SHOPPING_LIST_NAME};
    int[] toViews = {android.R.id.text1}; // The TextView in simple_list_item_1

    // Create an empty adapter we will use to display the loaded data.
    // We pass null for the cursor, then update it in onLoadFinished()
    mAdapter = new SimpleCursorAdapter(this.getActivity(), 
            android.R.layout.simple_list_item_1, null,
            fromColumns, toViews, 0);
    setListAdapter(mAdapter);

    // Prepare the loader.  Either re-connect with an existing one,
    // or start a new one.
    getLoaderManager().initLoader(0, null, this);
}

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    Log.d(LOG_TAG, "................. onCreateView()");
    return super.onCreateView(inflater, container, savedInstanceState);
}

// Called when a new Loader needs to be created
@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
    Log.d(LOG_TAG, "................. onCreateLoader()");
    // Now create and return a CursorLoader that will take care of
    // creating a Cursor for the data being displayed.
    return new CursorLoader(getActivity(), DbContentProvider.CONTENT_URI_SHOPPING_LISTS_META,
            PROJECTION, SELECTION, null, null);

}

// Called when a previously created loader has finished loading
@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
    Log.d(LOG_TAG, "................. onLoaderFinished()");
    // Swap the new cursor in.  (The framework will take care of closing the
    // old cursor once we return.)
    mAdapter.swapCursor(data);      
}

// Called when a previously created loader is reset, making the data unavailable
@Override
public void onLoaderReset(Loader<Cursor> loader) {
    Log.d(LOG_TAG, "................. onLoaderReset()");
    // This is called when the last Cursor provided to onLoadFinished()
    // above is about to be closed.  We need to make sure we are no
    // longer using it.
    mAdapter.swapCursor(null);      
}

@Override 
public void onListItemClick(ListView l, View v, int position, long id) {
    makeToast("shopping list clicked: " + position);
    Log.d(LOG_TAG, "shopping list clicked: " + position);
}

private void makeToast(String msg) {
    Toast.makeText(getActivity(), msg, Toast.LENGTH_SHORT).show();
}

}

注意 - 在我的 DbContentProvider 课程中,我有...

public static final Uri CONTENT_URI_SHOPPING_LISTS_META = Uri.parse("content://org.example.myapp.DbContentProvider/shopping_lists_meta"); public static final String SHOPPING_LISTS_META_TABLE = "shopping_lists_meta";

我的代码基于this Android ListFragment / LoaderManager example。而且我发现的与此类似的问题都没有提供解决我问题的解决方案。

我对 Android 还很陌生,所以这可能是我犯的一个简单错误。但是,本质上,在我看来,当在我的 DbContentProvider 类中调用 notifyChange() 时,我的 ListFragment 没有得到通知(或者该区域存在其他错误)。有人可以帮忙吗?

【问题讨论】:

    标签: java android android-listfragment simplecursoradapter android-loadermanager


    【解决方案1】:

    在花费数小时后,我创建了一个连接到本地联系人内容提供程序的 TestListActivity - 并且所有工作/更新都按应有的方式工作/更新,所以我知道问题可能出在我自己的内容提供程序中,我d 写的。

    我找到了答案here。原来我没有在我的内容提供者的query() 方法返回的cursor 上调用setNotificationUri(ContentResolver cr, Uri uri)。 (我确信这在我正在使用的 Reto Mauer 书中从未提到过......):/

    不管怎样,现在都整理好了! :)

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2011-09-16
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2010-12-31
      • 1970-01-01
      • 2020-02-12
      相关资源
      最近更新 更多