兔子--sqlite
在web开发中用惯了mysql,DB2等数据库,现在学习移动开发,接触到了SQLite这个轻型数据库,对移动开发的重要性不必多说,在学习的过程中发现一篇博文介绍的非常详细。在这里想与大家分享。 非常感谢原作者!以下是原文。 在Android中一共提供了5种数据存储方式,分别为: (1)Files:通过FileInputStream和FileOutputStream对文件进行操作。具体使用方法可以参阅博文《Android学习笔记34:使用文件存储数据》。 (2)SharedPreferences:常用来存储键值对形式的数据,对系统配置信息进行保存。具体使用方法可以参阅博文《Android学习笔记35:使用SharedPreferences方式存储数据》。 (3)ContentProviders:数据共享,用于应用程序之间数据的访问。 (4)SQLite:Android自带的轻量级关系型数据库,支持SQL语言,用来存储大量的数据,并且能够对数据进行使用、更新、维护等操作。 (5)Network:通过网络来存储和获取数据。 本篇博文介绍第四种方式,通过Android自带的SQLite数据库存储数据。
1.SQLite简介 SQLite是一款开源的、嵌入式关系型数据库,第一个版本Alpha发布于2000年。SQLite在便携性、易用性、紧凑性、高效性和可靠性方面有着突出的表现。 SQLite和C/S模式的数据库软件不同,它是一款嵌入式数据库,没有独立运行的进程,与所服务的应用程序在应用程序进程空间内共生共存。它的代码与应用程序代码也是在一起的,或者说嵌入其中,作为托管它的程序的一部分。因此不存在数据库的客户端和服务器,使用SQLite一般只需要带上它的一个动态库,就可以享受它的全部功能。 数据库服务器在程序中的好处是不需要网络配置或管理。将数据库客户端与服务器运行在同一个进程中,可以省去不少的操作及麻烦:不用担心防火墙或者地址解析;不用浪费时间管理复杂的授权和权限;可以减少网络调用相关的消耗;可以简化数据库管理并使程序更容易部署。 SQLite数据库通过数据库级上的独占性和共享锁来实现独立事务处理。这意味着多个进程可以在同一时间从同一数据库读取数据,但是只有一个可以写入数据。在某个进程向数据库执行写操作之前,必须获得独占锁定。在发出独占锁定后,其他的读写操作将不会再发生。 此外,SQLite数据库中的所有信息(比如表、视图、触发器等)都包含在一个文件内,方便管理和维护。SQLite数据库还支持大部分操作系统,除电脑上使用的操作系统之外,很多手机上使用的操作系统同样可以运行。同时,SQLite数据库还提供了多语言的编程接口,供开发者使用。 2.SQL基本命令 SQL是与关系型数据库通信的唯一方式。它专注于信息处理,是为构建、读取、写入、排序、过滤、映射、分组、聚集和通常的管理信息而设计的声明式语言。 在讲解SQL基本命令之前,有必要先了解一下SQLite所支持的数据类型都有哪些。 2.1SQLite支持的数据类型 SQLite采用动态数据存储类型,会根据存入的值自动进行判断。SQLite支持以下5种数据类型: (1)NULL:空值 (2)INTEGER:带符号的整型 (3)REAL:浮点型 (4)TEXT:字符串文本 (5)BLOB:二进制对象 2.2SQL基本命令 表是探索SQLite中SQL的起点,也是关系型数据库中信息的标准单位,所有的操作都是以表为中心的。那么如何使用SQL命令创建一张表呢? 2.2.1创建表 表是由行和列组成的,列称为字段,行称为记录。 使用CREATE命令可以创建表,CREATE命令的一般格式为: CREATE[TEMP/TEMPORARY]TABLEtable_name(column_definitions[,constraints]); 其中,[]中的内容是可选的,用TEMP或TEMPORARY关键字声明的表是临时表,这种表只存活于当前会话,一旦连接断开,就会被自动销毁。如果没有明确指出创建的表是临时表,则创建的是基本表,将会在数据库中持久存在,这也是数据库中最常见的表。 CREATETABLE命令至少需要一个表名和一个字段名,上述命令中的table_name表示表名,表名必须与其他标识符不同。column_definitions由用逗号分隔的字段列表组成,每个字段定义包括一个名称、一个域(类型)和一个逗号分隔的字段约束。其中,域是指存储在该列的信息的类型,约束用来控制什么样的值可以存储在表中或特定的字段中。 一条创建表的命令示例如下: 1 CREATE TABLE tab_student (studentId INTEGER PRIMARY KEY AUTOINCREMENT,2 studentName VARCHAR(20),128)">3 studentAge INTEGER); 如上,我们创建了一个名为tab_student的表,该表包含3个字段:studentId、studentName和studentAge,其数据类型分别为:INTEGER、VARCHAR和INTEGER。 此外,通过使用关键字PRIMARYKEY,我们指定了字段studentId所在的列是主键。主键确保了每一行记录在某种方式上与表中的其他行记录是不同的(唯一的),进而确保了表中的所有字段都是可寻址的。 SQLite为主键提供自增长功能,当定义字段类型为INTEGERPRIMARYKEY时,SQLite将为该字段创建默认值,该默认值确保整数值是唯一的。SQLite使用64-bit单符号整数主键,因此,该字段的最大值是9,223,372,036,854,775,807。当达到最大值时,SQLite会自动搜索该字段还未使用的值,并作为要插入的值。从表中删除记录时,rowid可能被回收并在后面的插入中使用。因此,新创建的rowid不一定是按照严格顺序增长的。如果想要SQLite使用唯一的自动主键值,而不是填补空白,可以在主键定义INTEGERPRIMARYKEY中加入关键字AUTOINCREMENT。AUTOINCREMENT关键字阻止rowid回收,它将为新插入的记录产生新的(不是回收的)rowid。 2.2.2插入记录 使用INSERT命令可以一次插入一条记录,INSERT命令的一般格式为: INSERTINTOtab_name(column_list)VALUES(value_list); 其中,tab_name指明将数据插入到哪个表中,column_list是用逗号分隔的字段名称,这些字段必须是表中存在的,value_list是用逗号分隔的值列表,这些值是与column_list中的字段一一对应的。 比如,向刚才创建的tab_student表中插入一条记录,便可以使用如下的语句完成: INSERT INTO tab_student (studentId,studentName,studentAge) VALUES (1,“jack”,23);通过以上的语句,便插入了一条studentName=”jack”,studentAge=”23”的记录,该记录的主键为studentId=1。 2.2.3更新记录 使用UPDATE命令可以更新表中的记录,该命令可以修改一个表中一行或者多行中的一个或多个字段。UPDATE命令的一般格式为: UPDATEtab_nameSETupdate_listWHEREpredicate; 其中,update_list是一个或多个字段赋值的列表,字段赋值的格式为column_name=value。WHERE子句使用断言识别要修改的行,然后将更新列应用到这些行。 比如,要更新刚才插入到tab_student表中的记录,便可以使用如下的语句完成: UPDATE tab_student SET studentName=”tom”,studentAge=”25” WHERE studentId=1;通过以上的语句,便可以将刚才插入的主键为studentId=1的记录更新为studentName=”tom”,studentAge=”25”了。 2.2.4删除记录 使用DELETE命令可以删除表中的记录,DELETE命令的一般格式为: DELETEFROMtable_nameWHEREpredicate; 其中,table_name指明所要删除的记录位于哪个表中。和UPDATE命令一样,WHERE子句使用断言识别要删除的行。 比如,要删除刚才插入的记录,便可以使用如下的语句完成: DELETE FROM tab_student 2.2.5查询记录 其中,通过第二个参数bindArgs,使SQL语句中的问号(?)与这个数组中的值形成一一对应关系,从而将值写入到“tab_student”表中的对应字段中。 6.2更新数据 更新数据的方法与添加数据的方法大致相同,具体如下: 1 /* 2 * Function : 更新数据 3 * 4 */ 5 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 } 6.3查询数据 查询数据时,因为需要返回查询的结果,所以需要使用SQLiteDatabase.rawQuery()方法将查询的结果返回,具体如下: 1 2 * Function : 查询数据 3 4 5 public Student findStudentInfo(int id) { 6 db = mySQLiteOpenHelper.getWritableDatabase(); 7 String sql = "SELECT studentId,studentAge FROM tab_student WHERE studentId = ?"; 8 Cursor cursor = db.rawQuery(sql,255)">new String[] {String.valueOf(id)}); 9 if(cursor.moveToNext()) { 10 return new Student(cursor.getInt(cursor.getColumnIndex("studentId")),cursor.getString(cursor.getColumnIndex("studentName")),128)">11 cursor.getInt(cursor.getColumnIndex("studentAge"))); 12 } 13 null; 14 }可以看出,通过使用SQLiteDatabase.rawQuery()方法可以将查询到的结果存入Cursor对象中。然后,我们可以使用Cursor对象的getXXX()方法将查询结果从Cursor对象中取出来。 当然了,我们还可以根据实际的需要,去实现更多的接口方法,比如,删除数据、获取数据列表、获取数据个数等等。 封装好了以上的这些接口方法,便可以很方便的在程序中直接调用这些方法,不必再去关心底层数据库的调用,而将精力放在UI界面的设计实现上。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |