Android SQLite数据库示例教程

时间:2020-02-23 14:29:18  来源:igfitidea点击:

欢迎使用Android SQLite示例教程。
Android SQLite是为Android应用程序存储数据的首选方法。
对于许多应用程序而言,无论直接使用SQLite还是通过某些第三方包装器,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 Device Monitor访问。

SQLite是一个典型的关系数据库,包含表(由行和列组成),索引等。
我们可以创建自己的表来相应地保存数据。
此结构称为架构。

Android SQLite SQLiteOpenHelper

Android具有可用于处理数据库模式更改的功能,这些功能主要取决于使用SQLiteOpenHelper类。

SQLiteOpenHelper旨在摆脱两个非常常见的问题。

  • 当应用程序首次运行时–此时,我们还没有数据库。
    因此,我们将不得不创建表,索引,启动器数据等。

  • 当应用程序升级到较新的架构时–我们的数据库仍将位于较旧版本的应用程序的旧架构上。
    我们将具有更改数据库架构的选项,以匹配应用程序其余部分的需求。

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

  • 构造函数:这需要Context(例如Activity),数据库名称,可选的游标工厂(我们将在后面讨论)和代表您使用的数据库模式版本的整数(通常从1开始)然后再增加)。

  • onCreate(SQLiteDatabase db):在没有数据库且应用需要一个数据库时调用。
    它向我们传递了一个SQLiteDatabase对象,该对象指向一个新创建的数据库,我们可以其中填充表和初始数据。

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

我们定义一个" DBManager"类来执行所有数据库CRUD(创建,读取,更新和删除)操作。

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

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

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

dbHelper是SQLiteOpenHelper子类的实例。

要关闭数据库连接,将调用以下方法。

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

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

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

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

内容值会使用给定的初始大小创建一组空值。
当我们跳到编码部分时,我们将讨论其他实例值。

更新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);
  }

Android SQLite –删除记录

我们只需要传递要删除的记录的ID,如下所示。

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游标

游标代表查询的整个结果集。
提取查询后,将对cursor.moveToFirst()进行调用。
调用moveToFirst()有两件事:

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

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

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

使用游标的另一种方法是将其包装在CursorAdapter中。
就像ArrayAdapter适应数组一样,CursorAdapter适应Cursor对象,使它们的数据可用于ListView之类的AdaptorView。

让我们跳到使用SQLite存储一些有意义的数据的项目。

Android SQLite示例项目结构

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

Android SQLite项目代码

该应用程序包含5个类。
我们首先使用DatabaseHelper进行定义,它是SQLiteOpenHelper的子类,如下所示:

DatabaseHelper.java

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;
  }

如上所述,除了构造函数之外,我们还覆盖了onCreate()和onUpgrade()方法。
我们已将名称分别分配给数据库和表theitroad_COUNTRIES.DB和COUNTRIES。
每当插入新行时,索引列都会自动递增。
国家和货币的列名称为"主题"和"描述"。

在DBManager类中,可以初始化DatabaseHelper并定义CRUD操作。
下面是该类的代码:

DBManager.java

package com.theitroad.sqlite;

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

public class DatabaseHelper extends SQLiteOpenHelper {

  //Table Name
  public static final String TABLE_NAME = "COUNTRIES";

  //Table columns
  public static final String _ID = "_id";
  public static final String SUBJECT = "subject";
  public static final String DESC = "description";

  //Database Information
  static final String DB_NAME = "theitroad_COUNTRIES.DB";

  //database version
  static final int DB_VERSION = 1;

  //Creating table query
  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);
  }
}

" CountryListActivity.java"类是在应用程序启动时启动的活动。
下面是为其定义的布局:

fragment_emp_list.xml

package com.theitroad.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);
  }

}

其中ListView组件被定义为包括存储在数据库中的记录。
最初,ListView将为空,因此TextView用于显示该内容。

CountryListActivity.java

<?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>

在此活动中,将调用DBManager对象以执行CRUD操作。

定义了一个SimpleCursorAdapter,以将游标对象中返回的查询结果中的元素添加到列表中。
在列表项上单击一个意图,以打开ModifyCountryActivity类。

该菜单包含一个用于从ActionBar添加新记录的项目。
在这里再次执行一个意图来打开AddCountryActivity类。
下面是menu.xml代码。

menu.xml

package com.theitroad.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);

      //OnCLickListiner For List Items
      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);
  }

}

" AddCountryActivity.java"文件的xml布局和代码定义如下:

activity_add_record.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>

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

AddCountryActivity.java

<?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>

此处执行的CRUD操作是将新记录添加到数据库中。

下面定义ModifyCountryActivity.java文件的xml布局和代码:

activity_modify_record.xml

package com.theitroad.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;
      }
  }

}

除了添加了"修改"和"删除"按钮之外,它与以前的布局相似。

ModifyCountryActivity.java

<?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>

此处执行的CRUD操作是更新和删除记录。

下图是我们项目最终输出的屏幕截图。

第一个图像是首次启动应用程序时看到的输出。

第二张图片是单击ActionBar中的菜单选项以添加新记录的结果,如下所示。

第三张图显示了添加3条记录时的输出:

第四个图像显示了单击任何列表项以修改或者删除记录时的输出:

最终图像是删除记录时的输出。
在这个例子中,我们删除第一条记录:

打开Android SQLite数据库文件

正如我们在本教程前面所讨论的,数据库文件存储在内部存储中,可从Android设备监控器访问该内部存储,如下图所示。

要查看此数据库,我们需要将该文件从设备中拉到桌面。
通过单击右上角的菜单选项来完成此操作,如下图所示:

要打开此文件,请从此链接下载SQLiteBrowser。

下面的片段显示了浏览器中的架构和表。

要查看该表,请转到顶部的"浏览数据"选项卡。