Android SQLite数据库示例教程

欢迎来到Android SQLite示例教程。Android SQLite是存储Android应用程序数据的首选方式。对于许多应用程序而言,SQLite是应用程序的支柱,无论是直接使用还是通过某些第三方封装。以下是我们今天将使用Android SQLite数据库创建的最终应用程序。

Android SQLite

Android SQLite是一个非常轻量级的数据库,内置于Android操作系统中。Android SQLite将清晰的SQL界面与非常小的内存占用和良好的速度结合在一起。对于Android,SQLite被”内嵌”到Android运行时中,因此每个Android应用程序都可以创建自己的SQLite数据库。Android SQLite本机API不是JDBC,因为JDBC可能对内存有限的智能手机来说开销太大。一旦成功创建数据库,它将位于data/data//databases/,可从Android设备监视器中访问。SQLite是典型的关系型数据库,包含表(由行和列组成)、索引等。我们可以创建自己的表来按照需要保存数据。这种结构被称为模式

Android SQLite SQLiteOpenHelper

Android 具有可用于处理数据库模式更改的功能,这主要取决于使用 SQLiteOpenHelper 类。SQLiteOpenHelper 的设计旨在解决两个非常常见的问题。

  1. 当应用程序第一次运行时 – 此时,我们还没有数据库。因此,我们将不得不创建表、索引、起始数据等。
  2. 当应用程序升级到新的模式时 – 我们的数据库仍将使用旧版应用程序的旧模式。我们将有选择地更改数据库模式以匹配应用程序的其余部分的需求。

SQLiteOpenHelper 封装了这些逻辑,以根据我们的规格创建和升级数据库。为此,我们需要创建 SQLiteOpenHelper 的自定义子类,至少实现以下三种方法。

  1. 构造函数:它接受上下文(例如,一个 Activity)、数据库的名称、可选的游标工厂(我们稍后会讨论此问题)和一个表示正在使用的数据库模式版本的整数(通常从 1 开始递增)。

    public DatabaseHelper(Context context) {
            super(context, DB_NAME, null, DB_VERSION);
        }
    
  2. onCreate(SQLiteDatabase db):当没有数据库且应用需要一个时调用。它向我们传递一个SQLiteDatabase对象,指向一个新创建的数据库,我们可以用表和初始数据填充它。

  3. onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion):当我们需要的架构版本与数据库的架构版本不匹配时调用,它向我们传递一个 SQLiteDatabase 对象和旧的以及新的版本号。因此,我们可以找出从旧架构转换到新架构的数据库的最佳方法。

我们定义了一个 DBManager 类来执行所有数据库的增删改查(CRUD)操作。

打开和关闭 Android SQLite 数据库连接

在执行任何数据库操作,如在表中插入、更新、删除记录之前,请通过调用以下方法打开数据库连接:getWritableDatabase(),如下所示:

public DBManager open() throws SQLException {
        dbHelper = new DatabaseHelper(context);
        database = dbHelper.getWritableDatabase();
        return this;
    }

dbHelperSQLiteOpenHelper子类的实例。要关闭数据库连接,请调用以下方法。

 public void close() {
        dbHelper.close();
    }

将新记录插入 Android SQLite 数据库表

以下代码片段显示了如何在 Android SQLite 数据库中插入新记录:

public void insert(String name, String desc) {
        ContentValues contentValue = new ContentValues();
        contentValue.put(DatabaseHelper.SUBJECT, name);
        contentValue.put(DatabaseHelper.DESC, desc);
        database.insert(DatabaseHelper.TABLE_NAME, null, contentValue);
    }

Content Values 使用给定的初始大小创建一个空值集。在我们深入编码部分时,我们将讨论其他实例值。

在 Android SQLite 数据库表中更新记录

以下代码片段展示了如何更新单个记录。

public int update(long _id, String name, String desc) {
        ContentValues contentValues = new ContentValues();
        contentValues.put(DatabaseHelper.SUBJECT, name);
        contentValues.put(DatabaseHelper.DESC, desc);
        int i = database.update(DatabaseHelper.TABLE_NAME, contentValues, DatabaseHelper._ID + " = " + _id, null);
        return i;
    }

Android SQLite – 删除记录

我们只需要像下面展示的那样传递要删除的记录的id。

public void delete(long _id) {
        database.delete(DatabaseHelper.TABLE_NAME, DatabaseHelper._ID + "=" + _id, null);
    }

Android SQLite Cursor

A Cursor represents the entire result set of the query. Once the query is fetched a call to cursor.moveToFirst() is made. Calling moveToFirst() does two things:

  • 它允许我们测试查询是否返回了空集(通过测试返回值)
  • 它将游标移动到第一个结果(当集合不为空时)

以下代码用于提取所有记录:

 public Cursor fetch() {
        String[] columns = new String[] { DatabaseHelper._ID, DatabaseHelper.SUBJECT, DatabaseHelper.DESC };
        Cursor cursor = database.query(DatabaseHelper.TABLE_NAME, columns, null, null, null, null, null);
        if (cursor != null) {
            cursor.moveToFirst();
        }
        return cursor;
    }

另一种使用游标的方法是将其包装在CursorAdapter中。就像ArrayAdapter适配数组一样,CursorAdapter适配Cursor对象,使它们的数据可用于像ListView这样的AdapterView。让我们跳转到我们使用SQLite存储一些有意义数据的项目中。

Android SQLite示例项目结构

在这个应用程序中,我们希望创建记录,以ListView的形式存储国家名称及其相应的货币。我们涵盖了上面讨论的所有功能。

Android SQLite 项目代码

该应用程序由 5 个类组成。我们首先定义DatabaseHelper,它是 SQLiteOpenHelper 的子类,如下所示:DatabaseHelper.java

package com.journaldev.sqlite;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;

public class DatabaseHelper extends SQLiteOpenHelper {

    // 表名
    public static final String TABLE_NAME = "COUNTRIES";

    // 表列
    public static final String _ID = "_id";
    public static final String SUBJECT = "subject";
    public static final String DESC = "description";

    // 数据库信息
    static final String DB_NAME = "JOURNALDEV_COUNTRIES.DB";

    // 数据库版本
    static final int DB_VERSION = 1;

    // 创建表查询
    private static final String CREATE_TABLE = "create table " + TABLE_NAME + "(" + _ID
            + " INTEGER PRIMARY KEY AUTOINCREMENT, " + SUBJECT + " TEXT NOT NULL, " + DESC + " TEXT);";

    public DatabaseHelper(Context context) {
        super(context, DB_NAME, null, DB_VERSION);
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL(CREATE_TABLE);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        db.execSQL("DROP TABLE IF EXISTS " + TABLE_NAME);
        onCreate(db);
    }
}

如上所述,我们已重写了onCreate()onUpgrade()方法,除了构造函数。我们将数据库和表命名为 JOURNALDEV_COUNTRIES.DB 和 COUNTRIES。索引列在插入新行时自动递增。国家和货币的列名分别为“subject”和“description”。DBManager 类是初始化 DatabaseHelper 并定义 CRUD 操作的地方。下面是这个类的代码:DBManager.java

package com.journaldev.sqlite;


import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;

public class DBManager {

    private DatabaseHelper dbHelper;

    private Context context;

    private SQLiteDatabase database;

    public DBManager(Context c) {
        context = c;
    }

    public DBManager open() throws SQLException {
        dbHelper = new DatabaseHelper(context);
        database = dbHelper.getWritableDatabase();
        return this;
    }

    public void close() {
        dbHelper.close();
    }

    public void insert(String name, String desc) {
        ContentValues contentValue = new ContentValues();
        contentValue.put(DatabaseHelper.SUBJECT, name);
        contentValue.put(DatabaseHelper.DESC, desc);
        database.insert(DatabaseHelper.TABLE_NAME, null, contentValue);
    }

    public Cursor fetch() {
        String[] columns = new String[] { DatabaseHelper._ID, DatabaseHelper.SUBJECT, DatabaseHelper.DESC };
        Cursor cursor = database.query(DatabaseHelper.TABLE_NAME, columns, null, null, null, null, null);
        if (cursor != null) {
            cursor.moveToFirst();
        }
        return cursor;
    }

    public int update(long _id, String name, String desc) {
        ContentValues contentValues = new ContentValues();
        contentValues.put(DatabaseHelper.SUBJECT, name);
        contentValues.put(DatabaseHelper.DESC, desc);
        int i = database.update(DatabaseHelper.TABLE_NAME, contentValues, DatabaseHelper._ID + " = " + _id, null);
        return i;
    }

    public void delete(long _id) {
        database.delete(DatabaseHelper.TABLE_NAME, DatabaseHelper._ID + "=" + _id, null);
    }

}

CountryListActivity.java类是应用程序启动时启动的活动。以下是为其定义的布局:fragment_emp_list.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="https://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent" >

    <ListView
        android:id="@+id/list_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:dividerHeight="1dp"
        android:padding="10dp" >
    </ListView>

    <TextView
        android:id="@+id/empty"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:text="@string/empty_list_text" />

</RelativeLayout>

在这里,定义了一个ListView组件,用于包含存储在数据库中的记录。最初,ListView将为空,因此使用TextView来显示相同的内容。CountryListActivity.java

package com.journaldev.sqlite;

import android.content.Intent;
import android.database.Cursor;
import android.os.Bundle;
import android.support.v4.widget.SimpleCursorAdapter;
import android.support.v7.app.ActionBarActivity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ListView;
import android.widget.TextView;

public class CountryListActivity extends ActionBarActivity {

    private DBManager dbManager;

    private ListView listView;

    private SimpleCursorAdapter adapter;

    final String[] from = new String[] { DatabaseHelper._ID,
            DatabaseHelper.SUBJECT, DatabaseHelper.DESC };

    final int[] to = new int[] { R.id.id, R.id.title, R.id.desc };

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

        setContentView(R.layout.fragment_emp_list);

        dbManager = new DBManager(this);
        dbManager.open();
        Cursor cursor = dbManager.fetch();

        listView = (ListView) findViewById(R.id.list_view);
        listView.setEmptyView(findViewById(R.id.empty));

        adapter = new SimpleCursorAdapter(this, R.layout.activity_view_record, cursor, from, to, 0);
        adapter.notifyDataSetChanged();

        listView.setAdapter(adapter);

        // 用于列表项的点击监听器
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView parent, View view, int position, long viewId) {
                TextView idTextView = (TextView) view.findViewById(R.id.id);
                TextView titleTextView = (TextView) view.findViewById(R.id.title);
                TextView descTextView = (TextView) view.findViewById(R.id.desc);

                String id = idTextView.getText().toString();
                String title = titleTextView.getText().toString();
                String desc = descTextView.getText().toString();

                Intent modify_intent = new Intent(getApplicationContext(), ModifyCountryActivity.class);
                modify_intent.putExtra("title", title);
                modify_intent.putExtra("desc", desc);
                modify_intent.putExtra("id", id);

                startActivity(modify_intent);
            }
        });
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {

        int id = item.getItemId();
        if (id == R.id.add_record) {

            Intent add_mem = new Intent(this, AddCountryActivity.class);
            startActivity(add_mem);

        }
        return super.onOptionsItemSelected(item);
    }

}

在此活动中,调用DBManager对象执行CRUD操作。定义了一个SimpleCursorAdapter,用于从查询结果中的Cursor对象向列表添加元素。在列表项点击时,执行意图打开ModifyCountryActivity类。菜单包含一个用于在ActionBar中添加新记录的项目。同样,在这里执行意图打开AddCountryActivity类。以下是menu.xml代码。menu.xml

<menu xmlns:android="https://schemas.android.com/apk/res/android"
    xmlns:app="https://schemas.android.com/apk/res-auto"
    xmlns:tools="https://schemas.android.com/tools"
    tools:context="com.example.sqlitesample.MainActivity" >

    <item
        android:id="@+id/add_record"
        android:icon="@android:drawable/ic_menu_add"
        android:orderInCategory="100"
        android:title="@string/add_record"
        app:showAsAction="always"/>

</menu>

AddCountryActivity.java文件的xml布局和代码如下:activity_add_record.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="https://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="20dp" >

    <EditText
        android:id="@+id/subject_edittext"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:ems="10"
        android:hint="@string/enter_title" >

        <requestFocus />
    </EditText>

    <EditText
        android:id="@+id/description_edittext"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:ems="10"
        android:hint="@string/enter_desc"
        android:inputType="textMultiLine"
        android:minLines="5" >
    </EditText>

    <Button
        android:id="@+id/add_record"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:text="@string/add_record" />

</LinearLayout>

定义了两个EditText组件,用于接收国家和货币的输入,以及一个按钮,用于将这些值添加到数据库并在ListView中显示。AddCountryActivity.java

package com.journaldev.sqlite;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;

public class AddCountryActivity extends Activity implements OnClickListener {

    private Button addTodoBtn;
    private EditText subjectEditText;
    private EditText descEditText;

    private DBManager dbManager;

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

        setTitle("Add Record");

        setContentView(R.layout.activity_add_record);

        subjectEditText = (EditText) findViewById(R.id.subject_edittext);
        descEditText = (EditText) findViewById(R.id.description_edittext);

        addTodoBtn = (Button) findViewById(R.id.add_record);

        dbManager = new DBManager(this);
        dbManager.open();
        addTodoBtn.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.add_record:

                final String name = subjectEditText.getText().toString();
                final String desc = descEditText.getText().toString();

                dbManager.insert(name, desc);

                Intent main = new Intent(AddCountryActivity.this, CountryListActivity.class)
                        .setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

                startActivity(main);
                break;
        }
    }

}

此处执行的CRUD操作是向数据库添加新记录。ModifyCountryActivity.java文件的xml布局和代码如下:activity_modify_record.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="https://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="10dp" >

    <EditText
        android:id="@+id/subject_edittext"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginBottom="10dp"
        android:ems="10"
        android:hint="@string/enter_title" />

    <EditText
        android:id="@+id/description_edittext"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:ems="10"
        android:hint="@string/enter_desc"
        android:inputType="textMultiLine"
        android:minLines="5" >
    </EditText>


    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:weightSum="2"
        android:gravity="center_horizontal"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/btn_update"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="@string/btn_update" />

        <Button
            android:id="@+id/btn_delete"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="@string/btn_delete" />
    </LinearLayout>

</LinearLayout>

与前一个布局类似,只是添加了修改和删除按钮。ModifyCountryActivity.java

package com.journaldev.sqlite;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;

public class ModifyCountryActivity extends Activity implements OnClickListener {

    private EditText titleText;
    private Button updateBtn, deleteBtn;
    private EditText descText;

    private long _id;

    private DBManager dbManager;

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

        setTitle("Modify Record");

        setContentView(R.layout.activity_modify_record);

        dbManager = new DBManager(this);
        dbManager.open();

        titleText = (EditText) findViewById(R.id.subject_edittext);
        descText = (EditText) findViewById(R.id.description_edittext);

        updateBtn = (Button) findViewById(R.id.btn_update);
        deleteBtn = (Button) findViewById(R.id.btn_delete);

        Intent intent = getIntent();
        String id = intent.getStringExtra("id");
        String name = intent.getStringExtra("title");
        String desc = intent.getStringExtra("desc");

        _id = Long.parseLong(id);

        titleText.setText(name);
        descText.setText(desc);

        updateBtn.setOnClickListener(this);
        deleteBtn.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.btn_update:
                String title = titleText.getText().toString();
                String desc = descText.getText().toString();

                dbManager.update(_id, title, desc);
                this.returnHome();
                break;

            case R.id.btn_delete:
                dbManager.delete(_id);
                this.returnHome();
                break;
        }
    }

    public void returnHome() {
        Intent home_intent = new Intent(getApplicationContext(), CountryListActivity.class)
                .setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        startActivity(home_intent);
    }
}

这里执行的 CRUD 操作是更新和删除记录。以下图片是我们项目的最终输出的屏幕截图。第一张图片是在首次启动应用程序时看到的输出。第二张图片是从操作栏中点击菜单选项添加新记录的结果,如下所示。第三张图片显示的是添加了 3 条记录后的输出:第四张图片显示的是点击任何列表项以修改或删除记录时的输出:最后一张图片是删除记录时的输出。在这个例子中,我们删除了第一条记录:

打开 Android SQLite 数据库文件

正如我们在本教程中之前讨论的那样,数据库文件存储在内部存储中,可以通过 Android 设备监视器访问,如下图所示。 要查看此数据库,我们需要将该文件从设备拉到我们的桌面上。这可以通过点击右上角的菜单选项来完成,如下图所示: 要打开此文件,请从这里链接下载 SQLiteBrowser。下面的片段显示了浏览器中的模式和表格。 要查看表格,请转到顶部的浏览数据选项卡。如下图所示: 这就结束了 Android SQLite 教程。最终的Android SQLite 项目可以从下面的链接下载。

下载 Android SQLite 示例项目

Source:
https://www.digitalocean.com/community/tutorials/android-sqlite-database-example-tutorial