1

我已经在网上浏览了 2 天,尝试了很多东西,但似乎无法弄清楚这有什么问题。

我对 Android 的开发还很陌生,所以我可能错过了一些明显的东西。

我有一个应用程序女巫正在使用 sqllite 数据库来存储一些数据,并为此概念证明在列表视图中显示它。我可以将项目添加到列表中,删除它们。

到目前为止,一切都很好。我遇到的问题是,当我没有删除更新数据库中名为“已删除”的列并将其设置为 1,然后让适配器更新列表时。它似乎不起作用。

如果我使用 delete 语句,它就可以工作。它更新了,一切都很好,但我想在数据库中有已删除的项目但不显示它们(所以基本上“隐藏”项目)

如果我检查数据库,更新本身成功了列更改和所有内容,所以我猜这是一个刷新问题,因为适配器不会重新查询数据库或那个方向的东西

列表视图加载器:

public void fillData() {

    if(lw.getAdapter() == null){
        // Fields from the database (projection)
        // Must include the _id column for the adapter to work
        String[] from = new String[] { TodoTable.COLUMN_SUMMARY, TodoTable.COLUMN_ID};
        String where = TodoTable.COLUMN_DELETED + " = ?";


        Cursor cursor = getContentResolver().query(TodoContentProvider.CONTENT_URI,from,where,new String[] {"0"},null);
        // Fields on the UI to which we map
        int[] to = new int[] { R.id.label };

        adapter = new SimpleCursorAdapter(this, R.layout.todo_row, cursor, from,
                to, 0);
        Log.v("Count",Integer.toString(cursor.getCount()));
        lw.setAdapter(adapter);
    }
    else
        adapter.notifyDataSetChanged();
    }

删除功能

@Override
public boolean onContextItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case DELETE_ID:
            /* Code for actual delete
            AdapterView.AdapterContextMenuInfo info = (AdapterView.AdapterContextMenuInfo) item
                    .getMenuInfo();
            Uri uri = Uri.parse(TodoContentProvider.CONTENT_URI + "/"
                    + info.id);
            getContentResolver().delete(uri, null, null);
            fillData();
            */

            /* Code for update and hide */
            AdapterView.AdapterContextMenuInfo info = (AdapterView.AdapterContextMenuInfo) item
                    .getMenuInfo();
            Uri uri = Uri.parse(TodoContentProvider.CONTENT_URI + "/"
                    + info.id);
            ContentValues values = new ContentValues();
            values.put(TodoTable.COLUMN_DIRTY, 1);
            values.put(TodoTable.COLUMN_DELETED, 1);
            getContentResolver().update(uri,values,null,null);
            fillData();
            return true;
    }
    return super.onContextItemSelected(item);
}

如果我将日志放到 ContentProvider 的查询函数中,它实际上不会触发。

关于如何解决这个问题的任何建议?

如果我使用adapter.swapCursor(cursor);它工作正常只是不知道这是否是这样做的正确方法。

public void fillData() {

    // Fields from the database (projection)
    // Must include the _id column for the adapter to work
    String[] from = new String[] { TodoTable.COLUMN_SUMMARY, TodoTable.COLUMN_ID};
    String where = TodoTable.COLUMN_DELETED + " = ?";

    Cursor cursor = getContentResolver().query(TodoContentProvider.CONTENT_URI,from,where,new String[] {"0"},null);

    // Fields on the UI to which we map
    int[] to = new int[] { R.id.label };

    if(lw.getAdapter() == null){
        adapter = new SimpleCursorAdapter(this, R.layout.todo_row, cursor, from,
                to, 0);
        Log.v("Count",Integer.toString(cursor.getCount()));
        lw.setAdapter(adapter);
    }
    else
    {
        adapter.swapCursor(cursor);
    }
}

泰来帮忙

4

2 回答 2

9

使用 adapter.swapCursor(cursor) 是正确的,因此您几乎可以回答自己的问题。

您的第一段代码不起作用,因为当您在数据库更新后调用 fillData() 时,您只需调用 adapter.notifyDataSetChanged() 并且数据集实际上并没有改变,因为光标是相同的。游标是对数据库中行的引用,更新基础数据库不会刷新游标。您的第二段代码确实刷新了光标并将新光标交换到适配器中(这也触发了对其绑定的视图的更新)。

更常见的编码方式是:

将此接口添加到您的活动中:

public class MyActivity extends Activity implementsLoaderManager.LoaderCallbacks<Cursor>

在onCreate中,设置适配器(注意此时光标为空):

String[] from = new String[] { TodoTable.COLUMN_SUMMARY, TodoTable.COLUMN_ID};
int[] to = new int[] { R.id.label };
adapter = new SimpleCursorAdapter(this, R.layout.todo_row, null, from, to, 0);  //Note that the cursor is null
lw.setAdapter(adapter);

启动加载器:

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

这会在后台线程中调用 onCreateLoader(因此,如果您的查询长时间运行,它不会阻塞 UI 线程)。完成后,在 UI 线程上调用 onLoadFinished,您可以在其中交换新光标。

执行删除或更新后,重新启动加载程序:

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

这将调用 onLoaderReset 从适配器中删除现有游标,然后再次调用 onCreateLoader 以换入新游标。

最后添加这些方法:

public Loader<Cursor> onCreateLoader(int id, Bundle args)
{
    String[] from = new String[] { TodoTable.COLUMN_SUMMARY, TodoTable.COLUMN_ID};
    String where = TodoTable.COLUMN_DELETED + " = ?";

    Loader<Cursor> loader = new CursorLoader(this, TodoContentProvider.CONTENT_URI, from, where, new String[] {"0"}, null);     
    return loader;
}

public void onLoadFinished(Loader<Cursor> loader, Cursor cursor)
{
    adapter.swapCursor(cursor);
}

public void onLoaderReset(Loader<Cursor> loader)
{
    adapter.swapCursor(null);
}
于 2013-10-29T11:53:25.267 回答
3

下面是我的工作解决方案。简而言之,我正在更新服务中的底层数据库,当服务完成其工作时,它会使用 localbroadcastmanager 调用活动。我使用 List 和 BaseAdapter。

在服务中,我调用:

    Intent intent = new Intent("notifyactivity");
    LocalBroadcastManager.getInstance(this).sendBroadcast(intent);

在活动中:

    @Override
        public void onResume() {
        super.onResume();
       LocalBroadcastManager.getInstance(this).registerReceiver(mMessageReceiver,new IntentFilter("notifyactivity"));
}

private BroadcastReceiver mMessageReceiver = new BroadcastReceiver() {
  @Override
  public void onReceive(Context context, Intent intent) {         
      applicationcontacts.clear(); //clear the list first
      applicationcontacts.addAll(db.getAllContacts()); //reload the list
      listview=(ListView) findViewById(R.id.listview1);
      listview.setAdapter(listadaptor);
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                listadaptor.notifyDataSetChanged();
            }
        });
  }
};

@Override
protected void onPause() {
  LocalBroadcastManager.getInstance(this).unregisterReceiver(mMessageReceiver);
  super.onPause();
}
于 2015-04-17T07:38:04.857 回答