首页 > 代码库 > 使用Sqlite数据库存储数据
使用Sqlite数据库存储数据
1.Sql基本命令
1.1.创建表
表是有行和列组成的,列称为字段,行称为记录。
使用CREATE命令来创建表:
1 CREATE TABLE tab_student (studentId INTEGER PRIMARY KEY AUTOINCREMENT, 2 studentName VARCHAR(20), 3 studentAge INTEGER);
1.2.插入记录(行)
使用INSERT命令可以一次插入一条记录,INSERT命令的一般格式为:
INSERT INTO tab_student (studentId, studentName, studentAge) VALUES (1, “jack”, 23);
1.3.更新记录(行)
使用UPDATE命令可以更新表中的记录,该命令可以修改一个表中一行或者多行中的一个或多个字段。UPDATE命令的一般格式为:
UPDATE tab_student SET studentName=”tom”, studentAge=”25” WHERE studentId=1;
1.4.删除记录(行)
使用DELETE命令可以删除表中的记录,DELETE命令的一般格式为:
DELETE FROM tab_student WHERE studentId=1;
1.5.查询记录(行)
SELECT命令是查询数据库的唯一命令。SELECT命令也是SQL命令中最大、最复杂的命令。
SELECT命令的通用形式如下:
SELECT [distinct] heading
FROM tables
WHERE predicate
GROUP BY columns
HAVING predicate
ORDER BY columns
LIMIT count,offset;
其中,每个关键字(如FROM、WHERE、HAVING等)都是一个单独的子句,每个子句由关键字和跟随的参数构成。GROUP BY和HAVING一起工作可以对GROUP BY进行约束。ORDER BY使记录集在返回之前按一个或多个字段的值进行排序,可以指定排序方式为ASC(默认的升序)或DESC(降序)。此外,还可以使用LIMIT限定结果集的大小和范围,count指定返回记录的最大数量,offset指定偏移的记录数。
在上述的SELECT命令通用形式中,除了SELECT之外,所有的子句都是可选的。目前最常用的SELECT命令由三个子句组成:SELECT、FROM、WHERE,其基本语法形式如下:
SELECT heading FROM tables WHERE predicate;
比如,要查询刚才插入的记录,便可以使用如下的语句完成:
SELECT studentId, studentName, studentAge FROM tab_student WHERE studentId=1;
2.数据库操作辅助类SQLiteOpenHelper
Android提供了一个重要的类SQLiteOpenHelper,用于辅助用户对SQLite数据库进行操作。
SQLiteOpenHelper的构造函数原型如下:
public SQLiteOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version);
其中,参数context表示应用程序运行的环境,包含应用程序所需的共享资源。参数name表示Android的数据库名字。参数factory是SQLiteDatabase.CursorFactory类对象,用于存储查询Android SQLite数据库的结果集。参数version表示应用程序所用的数据库的版本,该版本并非SQLite的真正版本,而是指定应用程序中的SQLite数据库的版本,当该版本号发生变化时,将会触发SQLiteOpenHelper类中的onUpgrade()或onDowngrade()方法。
SQLiteOpenHelper类的所有方法如图1所示。
其中,close()方法用于关闭SQLiteOpenHelper对象中的SQLite数据库;getReadableDatabase()方法和getWriteableDatabase()方法类似,getReadableDatabase()方法以只读状态打开SQLiteOpenHelper对象中指定的SQLite数据库,任何想要修改数据库的操作都是不允许的;getWriteableDatabase()方法也是打开数据库,但是允许数据库正常的读/写操作;在一个不存在的数据库上调用任何方法时,都会隐式的调用SQLiteOpenHelper对象的onCreate()方法;当应用程序第一次访问数据库时,则会调用onOpen()方法,但是,如果版本号发生了变化的话,则会调用onUpgrade()或onDowngrade()方法。
3.数据库类SQLiteDatabase
SQLiteDatabase类用来完成对数据库的操作任务,比如表的选择、插入、更新和删除语句等。
SQLiteDatabase类中常用的用于执行SQL语句的方法有以下一些。
(1)execSQL()方法:
public void execSQL (String sql);
public void execSQL (String sql, Object[] bindArgs);
(2)query()方法:
public Cursor query (String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having,String orderBy, String limit);
public Cursor query (boolean distinct, String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit, CancellationSignal cancellationSignal);
public Cursor query (String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having,String orderBy);
public Cursor query (boolean distinct, String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);
(3)queryWithFactory()方法:
public Cursor queryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, boolean distinct, String table, String[]columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit,CancellationSignal cancellationSignal);
public Cursor queryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, boolean distinct, String table, String[]columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);
(4)rawQuery()方法:
public Cursor rawQuery (String sql, String[] selectionArgs, CancellationSignal cancellationSignal);
public Cursor rawQuery (String sql, String[] selectionArgs);
(5)rawQueryWithFactory()方法:
public Cursor rawQueryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, String sql, String[] selectionArgs,String editTable);
public Cursor rawQueryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, String sql, String[] selectionArgs,String editTable, CancellationSignal cancellationSignal);
其中,execSQL()方法都有一个参数sql,这个参数是一个SQL语句。第二个参数bindArgs接收一个数组,数组中的每个成员捆绑了一个查询。execSQL()方法用于运行那些没有返回值的查询语句,比如创建、插入、更新和修改表。
query()方法和queryWithFactory()方法是在数据库中运行一些轻量级的单查询语句,参数包括table、columns、groupBy、having、orderBy、limit等SQL语句关键字。这些方法允许将SQL语句传递给相关方法,而不必直接使用SQL语句。
rawQuery()方法和rawQueryWithFactory()方法也都有一个参数sql,用于执行SQL查询语句,返回值是Cursor对象。这两个方法都有一个版本能够接收一个字符串数组selectionArgs作为参数,通过这个参数,SQLiteDatabase对象将把捆绑的SQL语句中的问号(?)用这个数组中的值代替,并按照一一对应的位置关系进行取代。
SQLiteDatabase类提供了大约50个方法,除此之外还有一些用于打开数据库的方法(如openDatabase()、openOrCreateDatabase()等),用于管理SQLite事务的方法(如beginTransaction()、endTransaction()等),用于测试数据库是否被锁住的方法(如isDbLockedByCurrentThread()、isDbLockedByOtherThread()等),以及获取数据库基本信息的方法(如getMaximumSiza()、getVersion()等)。这里就不一一介绍了,具体可以参阅SQLiteDatabase类的API帮助文档。
4.游标类Cursor
在Android中,查询数据是通过Cursor类来实现的,当我们使用SQLiteDatabase.query()或SQLiteDatabase.rawQuery()方法时,会得到一个Cursor对象,Cursor指向的就是每一条记录,它提供了很多有关查询的方法,如图2所示。
5.封装接口
有了以上的基础,我们便可以按照MVC的架构,封装一个接口层,在该接口层中实现对SQLite数据库的具体操作。
以下分别以添加数据、更新数据、查询数据为例讲解其具体的实现方法。在实现这些方法之前,我们首先需要创建一张表。这里我创建了一个名为MySQLiteOpenHelper的类,让它继承自SQLiteOpenHelper类,并实现了SQLiteOpenHelper类的onCreate()方法,在该方法里实现创建一张表的操作,具体源代码如下:
1 /* 2 * Function : 创建表 3 * Author : 博客园-依旧淡然 4 */ 5 public void onCreate(SQLiteDatabase db) { 6 db.execSQL("CREATE TABLE tab_student (studentId INTEGER PRIMARY KEY AUTOINCREMENT, " + "studentName VARCHER(20), " + "studentAge INTEGER)"); 7 }
通过以上的代码,我们创建了一张名为“tab_student”的表,并在该表中创建了三个字段,分别为:studentId、studentName和studentAge。并且指定了studentId字段作为该表的主键。
5.1添加数据
添加数据可以使用SQLiteDatabase.execSQL(String sql, Object[] bindArgs)方法来实现,具体如下:
1 /* 2 * Function : 添加数据 3 * Author : 博客园-pres_cheng 4 */ 5 public void addStudentInfo(Student student) { 6 db = mySQLiteOpenHelper.getWritableDatabase(); 7 db.execSQL("INSERT INTO tab_student (studentId, studentName, studentAge) values (?, ?, ?)", 8 new Object[] {student.getStudentId(), student.getStudentName(), student.getStudentAge()}); 9 }
其中,通过第二个参数bindArgs,使SQL语句中的问号(?)与这个数组中的值形成一一对应关系,从而将值写入到“tab_student”表中的对应字段中。
5.2更新数据
更新数据的方法与添加数据的方法大致相同,具体如下:
1 /* 2 * Function : 更新数据 3 * Author : 博客园-pres_cheng 4 */ 5 public void updateStudentInfo(Student student) { 6 db = mySQLiteOpenHelper.getWritableDatabase(); 7 db.execSQL("UPDATE tab_student SET studentName = ?, studentAge = ? WHERE studentId = ?", new Object[] {student.getStudentName(), student.getStudentAge(), student.getStudentId()}); 8 }
5.3查询数据
查询数据时,因为需要返回查询的结果,所以需要使用SQLiteDatabase.rawQuery()方法将查询的结果返回,具体如下:
1 /* 2 * Function : 查询数据 3 * Author : 博客园-pres_cheng 4 */ 5 public Student findStudentInfo(int id) { 6 db = mySQLiteOpenHelper.getWritableDatabase(); 7 String sql = "SELECT studentId, studentName, studentAge FROM tab_student WHERE studentId = ?"; 8 Cursor cursor = db.rawQuery(sql, new String[] {String.valueOf(id)}); 9 if(cursor.moveToNext()) { 10 return new Student(cursor.getInt(cursor.getColumnIndex("studentId")), cursor.getString(cursor.getColumnIndex("studentName")), 11 cursor.getInt(cursor.getColumnIndex("studentAge"))); 12 } 13 return null; 14 }
可以看出,通过使用SQLiteDatabase.rawQuery()方法可以将查询到的结果存入Cursor对象中。然后,我们可以使用Cursor对象的getXXX()方法将查询结果从Cursor对象中取出来。
当然了,我们还可以根据实际的需要,去实现更多的接口方法,比如,删除数据、获取数据列表、获取数据个数等等。
封装好了以上的这些接口方法,便可以很方便的在程序中直接调用这些方法,不必再去关心底层数据库的调用,而将精力放在UI界面的设计实现上。
使用Sqlite数据库存储数据