【问题标题】:Android: Using SimpleCursorAdapter to get Data from Database to ListViewAndroid:使用 SimpleCursorAdapter 从数据库获取数据到 ListView
【发布时间】:2012-08-22 16:59:15
【问题描述】:

我正在编写一个应该使用数据库来存储数据并从中读取数据的 android 应用程序。使用this tutorial (on archive.org),我得到了创建数据库的应用程序,并且能够创建新条目,但是,我不知道如何读取数据库以获取存储在 ListView 中的数据。我知道这个网站上有很多类似的问题,但似乎没有一个适用于教程中的数据库的方式。

代码:

import java.util.Calendar;

import maturarbeit.nicola_pfister.studenttools.database.DBAdapter;
import android.app.AlertDialog.Builder;
import android.app.ListActivity;
import android.content.DialogInterface;
import android.content.DialogInterface.OnClickListener;
import android.database.Cursor;
import android.os.Bundle;
import android.support.v4.widget.SimpleCursorAdapter;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import android.widget.ListView;


public class Marks extends ListActivity {

DBAdapter db = new DBAdapter(this);

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.marks);
}

@Override
protected void onPause() {
    super.onPause();
    db.close();
}

@Override
protected void onResume() {
    super.onResume();
    db.open();

    getData();
}

@SuppressWarnings("deprecation")
private void getData() {
    SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, 
            android.R.layout.simple_list_item_1, 
            db.getAllMarks(), 
            new String[] { "value" }, 
            new int[] { android.R.id.text1 });

    ListView listView = (ListView) findViewById(R.id.marks_list);
    listView.setAdapter(adapter);
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.marks, menu);
    return true;
}

public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
    case R.id.menu_add:
        Calendar cal = Calendar.getInstance();
        int day = cal.get(Calendar.DAY_OF_MONTH);
        int month = cal.get(Calendar.MONTH);
        final String date = day + "." + month;
        Builder builder = new Builder(this);
        builder
            .setTitle(R.string.dialog_addmarks)
            .setItems(R.array.markslist, new OnClickListener() {
                public void onClick(DialogInterface dialog, int which) {
                    @SuppressWarnings("unused")
                    long id;
                    String selection = getResources().getStringArray(R.array.markslist)[which];
                    id = db.insertMark(date, "Default", selection);
                    }
            })
            .show();
        getData();
        break;
    case R.id.menu_delete:
            //Deleting function yet to be implemented.
        break;
    }
    return super.onOptionsItemSelected(item);
}
}

编辑:ListView ID 错误,因为它必须是 android:list。

【问题讨论】:

    标签: android database listview simplecursoradapter


    【解决方案1】:

    使用您链接到的教程中的数据库格式,每一行都有一个_idisbntitlepublisher。现在让我们假设您想在 ListView 中显示每个标题:

    db = new DBAdapter(this);
    SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, 
            android.R.layout.simple_list_item_1, 
            db.getAllTitles(), 
            new String[] { "title" }, 
            new int[] { android.R.id.text1 });
    
    ListView listView = (ListView) findViewById(R.id.list);
    listView.setAdapter(adapter);
    

    (您不需要自己循环遍历光标,适配器会为您完成这项工作!)

    您的 SimpleCursorAdapter 构造函数中的最后两个参数是您缺少的。它们是“from”和“to”参数:

    • 我们想要获取存储在列名title 中的每本书的名称,这是我们获取“来自”信息的地方。
    • 接下来我们需要告诉它“去”到哪里:android.R.id.text1android.R.layout.simple_list_item_1 布局中的 TextView。 (您可以自己挖掘您的 SDK 并查看 simple_list_item_1.xml 文件,或者暂时相信我。)

    现在“from”和“to”参数都是数组,因为我们可以传递不止一列的信息,也可以试试这个适配器:

    SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, 
            android.R.layout.simple_list_item_2, 
            db.getAllTitles(), 
            new String[] { "title", "publisher" }, 
            new int[] { android.R.id.text1, android.R.id.text2 });
    

    使用此适配器,其数据库中的书籍将按标题显示,然后按出版商显示。我们所要做的就是使用一个布局android.R.layout.simple_list_item_2,它接受两个字段并定义哪些列转到哪些TextView。

    我希望这会有所帮助。还有很多东西要学,但也许这会给你一些基础知识。


    最后评论

    在我的脑海中,添加新数据后刷新 ListView 试试这个:

    public void onClick(DialogInterface dialog, int which) {
        String selection = getResources().getStringArray(R.array.markslist)[which];
        db.insertMark(date, "Default", selection);
        cursor.requery();
        adapter.notifyDataSetChanged();
    }
    

    您必须定义adapter 并为cursor 创建一个变量,但这很简单:

    public class Marks extends ListActivity {
        SimpleCursorAdapter adapter;
        Cursor cursor;
        DBAdapter db = new DBAdapter(this);
        ...
    

    并相应地更改 getData():

    private void getData() {
        cursor = db.getAllMarks();
        adapter = new SimpleCursorAdapter(this, 
                android.R.layout.simple_list_item_1, 
                cursor, 
                new String[] { "value" }, 
                new int[] { android.R.id.text1 });
        ...
    }
    

    祝你好运!

    【讨论】:

    • 谢谢。我想我明白了,你想解释什么。我使用了你的适配器,但我仍然遇到一个异常,导致我的应用程序崩溃。我认为错误是在其他地方吗?
    • 复制并粘贴 LogCat(使用问题左下角的编辑链接)。我会看看。完成此操作后,在评论中标记我(使用“@Sam”)。
    • 我将 LogCat 添加到我的问题中。谢谢你帮助我。
    • 嗯,我不确定发生了什么。看起来您的调试器从未连接到您的应用程序并超时:Launch timeout has expired, giving up wake lock!。您是否尝试过在没有调试器的情况下运行应用程序? (在 Eclipse 中,Ctrl+F11)
    • 此处使用的方法在 API 级别 11 中已弃用。developer.android.com/reference/android/widget/…, int, android.database.Cursor, java.lang.String[], int[])
    【解决方案2】:

    不是在游标中的每个项目上都使用适配器,而是为整个游标创建一个适配器。您可以设置此列表视图以使用该游标。尝试一些 SimpleCursorAdapter 教程,例如 this

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2013-07-20
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2020-03-26
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多