很多时候我们的APP都不能独立于数据之外,常常对数据进行各种操作。Android为数据存储提供了多种方式,主要有五种:文件、SharedPreferences、SQLite、网络、内容提供者。这一篇先总结下关于如何使用SQLite数据库。
在Android平台上,集成了一个嵌入式关系型数据库—SQLite,SQLite3支持 NULL、INTEGER、REAL(浮点数字)、TEXT(字符串文本)和BLOB(二进制对象)数据类型,虽然它支持的类型虽然只有五种,但实际上sqlite3也接受varchar(n)、char(n)、decimal(p,s) 等数据类型,只不过在运算或保存时会转成对应的五种数据类型。 SQLite最大的特点是你可以保存任何类型的数据到任何字段中,无论这列声明的数据类型是什么,也可以不必在创建表的时候指定字段的长度,但是作为主键的字段必须是Integer类型的。例如:可以在Integer字段中存放字符串,或者在布尔型字段中存放浮点数等。 但有一种情况例外:定义为INTEGER PRIMARY KEY的字段只能存储64位整数, 当向这种字段中保存除整数以外的数据时,将会产生错误。 另外, SQLite 在解析CREATE TABLE 语句时,会忽略 CREATE TABLE 语句中跟在字段名后面的数据类型信息,又如下面语句会忽略 name字段的类型信息:
CREATE TABLE payout (_id integer primary key autoincrement, name varchar(20))其他相关知识欲了解更多,请参见SQlite官网
SQLite数据库和其他关系型数据一样,也一样支持数据的查询、插入、更新、删除、视图、索引、触发器、连接、聚合函数等常见操作。SQl语法形式和MSSQL、Oracle大同小异。其中中括号代表可选项,非必填。
create if not exists table 表名(字段列表 [default 默认值]);
create table if not exists 表名(字段列表 [check(布尔表达式)]);
create table if not exists 表名(字段列表 [unique]);
alter table 旧表名 rename to 新表名;
alter table 表名 add column 新字段名 类型;
select * from 表名 [where 条件子句 group by 分组字句 having ... order by 排序子句] //如:select name from payout group by name having count(*)>1
insert into 表名(字段列表) values(值列表) //如: insert into payout(name, fee) values('旅游',2000)
update 表名 set 字段名=值 [where 条件子句] //如:update payout set fee=2800 where _id=10
delete from 表名 [where 条件子句] //如:delete from payout where id=10
create view if not exists 视图名 as sql查询语句结果集合 /*CREATE VIEW if not exists user_count AS SELECT count(user_id) FROM *user;
*应用:SELECT * FROM user_count */
create temp view 视图名 as sql查询语句结果集合
drop view if exists 视图名 //如:drop view if exists user_count;
create index 索引名 on 数据表(字段);
create index 索引名 on 数据表(字段名1 asc, 字段2 desc);
create unique index 索引名 on 数据表(字段名1 asc, 字段2 desc);
drop index if exists 索引名;
select xxx from 表名1 inner join 表名2 on 表名1.列名 = 表名2.列名
select xxx from 表名1 left outer join 表名2 on 表名1.列名 = 表名2.列名
Android中集成了SQlite自然也就封装了许多操作数据库的类,掌握以下几个类就能在Android应用中熟练使用SQlite。
在用户初次使用APP时,需要创建应用使用到的数据库表结构及添加一些初始化记录,当然也能在APP升级的时候,也需要对数据表结构进行更新。在Android系统,封装了一个名为SQLiteOpenHelper的抽象类,该类用于对数据库版本进行管理,里面封装了一些方法用户创建数据库、数据表、获取数据库的实例等。
当调用SQLiteOpenHelper的getWritableDatabase()或者getReadableDatabase()方法获取用于操作数据库的SQLiteDatabase实例的时候,如果数据库不存在,Android系统会自动生成一个数据库,接着调用onCreate()方法,需要注意的是,getWritableDatabase()和getReadableDatabase()方法都可以获取一个用于操作数据库的SQLiteDatabase实例。但getWritableDatabase() 方法以读写方式打开数据库,一旦数据库的磁盘空间满了,数据库就只能读而不能写,倘若使用的是getWritableDatabase() 方法就会出错。getReadableDatabase()方法先以读写方式打开数据库,如果数据库的磁盘空间满了,就会打开失败,当打开失败后会继续尝试以只读方式打开数据库。
onCreate()方法在初次生成数据库时才会被调用,在onCreate()方法里可以生成数据库表结构及添加一些应用使用到的初始化数据。onUpgrade()方法在数据库的版本发生变化时会被调用,数据库的版本是由程序员控制的,假设数据库现在的版本(是由创建数据库时第三个参数值决定的)是1,由于业务的需要,修改了数据库表的结构,这时候就需要升级软件,升级软件时希望更新用户手机里的数据库表结构,为了实现这一目的,可以把原来的数据库版本设置为2(当然3…任意整数,只要不为0即可)
该类封装了一些操作数据库的API,使用该类可以完成对数据进行添加(Create)、查询(Retrieve)、更新(Update)和删除(Delete)操作(这些操作简称为CRUD)。SQLiteOpenHelper的作用之一就是为了得到这个实例,SQLiteDatabase类也支持(占位符?)。
private final String createStudTb="Create Table Student(_id integer primary key autoincrement,name varchar(16),sex varchar(2))"; //创建数据表Student arg0.execSQL(createStudTb);
Cursor cursor = db.rawQuery("select * from Student where name like ?", new String[]{"%cm%"});
@param table:数据表名
@param nullColumnHack:(SQL doesn't allow inserting a completely empty row, so if initialValues is empty this column will explicitly be assigned a NULL value
)如果第三个参数values 为Null或者元素个数为0, Insert()方法必然要添加一条除了主键之外其它字段为Null值的记录,为了满足这条insert语句的语法, insert语句必须给定一个字段名,如:insert into payout(name) values(NULL),倘若不给定字段名 , insert语句就成了这样: insert into payout() values(),显然这不满足标准SQL的语法。对于字段名,建议使用主键之外的字段,如果使用了INTEGER类型的主键字段,执行类似insert into payout(_id) values(NULL)的insert语句后,该主键字段值也不会为NULL。如果第三个参数values 不为Null并且元素的个数大于0 ,可以把第二个参数设置为null。 @param values:要保存的键值对对象 @return:返回新插入记录的rowid,插入失败返回-1
@param whereClause:Where子句,不能包含where关键句
@param whereArgs: 替换占位符的实参
@return 返回删除的行数
@param table:表名。相当于select语句from关键字后面的部分。如果是多表联合查询,可以用逗号将两个表名分开。 @param columns:要查询出来的列名。相当于select语句select关键字后面的部分。 @param selection:查询条件子句,相当于select语句where关键字后面的部分,在条件子句允许使用占位符“?” @param selectionArgs:对应于selection语句中占位符的值,值在数组中的位置与占位符在语句中的位置必须一致,否则就会有异常。 @param groupBy:相当于select语句group by关键字后面的部分 @param having:相当于select语句having关键字后面的部分 @param orderBy:相当于select语句order by关键字后面的部分,如:_id desc, age asc;
@param limit:指定偏移量和获取的记录数,相当于select语句limit关键字后面的部分。
查询结果的返回集合一般都是以Cursor形式放回的,和其他数据库的游标特点大同小异。
ContentValues 和 HashTable 类似都是一种存储的机制,但是两者最大的区别就在于:ContentValues 只能存储基本类型的数据,像string、int之类的,不能存储对象这种东西,而HashTable却可以存储对象。操作方法和HasMap大同小异。
public void saveByInsert(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
ContentValues values=new ContentValues();
values.put("name", stu.getName());
values.put("sex",stu.getSex());
db.insert("Student", null, values);
db.close();
}
package cmo.learn.service;
import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
public class DBHelper extends SQLiteOpenHelper {
private static final String dbname="STUDENT.db";
private static final int version=1;
private static DBHelper dbHelper;
private final String createStudTb="Create Table Student(_id integer primary key autoincrement,name varchar(16),sex varchar(2))";//可以不指定字段的类型、长度,因为int类型也可以保存Char类型的创建学生表
/** * SQLiteOpenHelper(context, name, factory, version) * @param context 上下文对象 * @param name 数据库名称 * @param factory 游标工厂对象,为null则取系统默认的游标工厂对象 * @param version 数据库版本号取值为任意非0整数 */
public DBHelper(Context context) {
//super(context, dbname, null, version);//创建DB成功之后会保存在/<package name>/databases/初始版本号为1
super(context, dbname, null, 2);
}
public static DBHelper getInstance(Context context) {
if (dbHelper == null) { //单例模式
dbHelper = new DBHelper(context.getApplicationContext());
}
return dbHelper;
}
@Override
public void onCreate(SQLiteDatabase arg0) {
//当数据库第一次创建时触发,所以可以在这可以创建数据库表和完成一些初始化操作,SQLiteDatabase封装了针对SQLite的所有操作,数据库操作实例类
arg0.execSQL(createStudTb);
}
@Override
public void onUpgrade(SQLiteDatabase arg0, int arg1, int arg2) {
//当version值与初始值不同时触发
arg0.execSQL("Alter Table Student add amount integer");
}
}
package cmo.learn.service;
import java.util.ArrayList;
import java.util.List;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import cmo.learn.domain.Student;
public class StudentDBOperate {
private DBHelper dbhelper;
//要操作数据库操作实例首先得得到数据库操作实例
public StudentDBOperate(Context context){
this.dbhelper=new DBHelper(context);
}
public void save(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Insert into Student(name,sex) Values(?,?)",
new Object[] { stu.getName(),stu.getSex() });
db.close();
}
public void saveByInsert(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
ContentValues values=new ContentValues();
values.put("name", stu.getName());
values.put("sex",stu.getSex());
db.insert("Student", null, values);
db.close();
}
public void delete2(Integer id){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.delete("Student", "_id=?", new String[]{id.toString()});
}
public void delete(Integer id){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Delete from Student where _id=?",
new String[] { String.valueOf(id) });
db.close();
}
public void update2(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
ContentValues values=new ContentValues();
values.put("name", stu.getName());
values.put("sex",stu.getSex());
db.update("Student", values, "_id=?", new String[]{String.valueOf(stu.get_id())});
}
public void update(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Update Student Set name=?,sex=? where _id=?",
new String[] {stu.getName(),stu.getSex(),String.valueOf(stu.get_id()) });
db.close();
}
public Student query(Integer _id){
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf=db.rawQuery("Select _id,name,sex from Student where _id=?",
new String[] {String.valueOf(_id)});
if(stuimf!=null){
if(stuimf.moveToFirst()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
return new Student(id,name,sex);
}
}
db.close();
return null;
}
public Student query2(Integer _id){
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf= db.query("Student", new String[]{"_id","name","sex"}, "_id=?",new String[]{_id.toString()}, null, null, null);
if(stuimf!=null){
if(stuimf.moveToFirst()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
return new Student(id,name,sex);
}
}
return null;
}
/** * 分页查询 * @param offset 跳过前面多少记录 * @param maxRecord 每页获取多少记录 * @return */
public List<Student> getScrollData(int offset,int maxRecord){
SQLiteDatabase db=dbhelper.getReadableDatabase();
List<Student> stulist = new ArrayList<Student>();
Cursor stuimf=db.rawQuery("Select _id,name,sex from Student order by _id asc limit ?,?",
new String[] {String.valueOf(offset),String.valueOf(maxRecord)});
if(stuimf!=null){
while(stuimf.moveToNext()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
stulist.add(new Student(id,name,sex));
}
return stulist;
}
db.close();
return null;
}
public long getCount(){
long result=0;
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf=db.rawQuery("Select count(*) from Student", null);
if(stuimf!=null){
//因为只要创建了表,count(*)最小值为1,即还未插入数据时
stuimf.moveToFirst();
result=stuimf.getLong(0);
}
return result;
}
}
package cmo.learn.service;
import java.util.ArrayList;
import java.util.List;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import cmo.learn.domain.Student;
public class StudentDBOperate {
private DBHelper dbhelper;
//要操作数据库操作实例首先得得到数据库操作实例
public StudentDBOperate(Context context){
this.dbhelper=new DBHelper(context);
}
public void save(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Insert into Student(name,sex) Values(?,?)",
new Object[] { stu.getName(),stu.getSex() });
db.close();
}
public void saveByInsert(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
ContentValues values=new ContentValues();
values.put("name", stu.getName());
values.put("sex",stu.getSex());
db.insert("Student", null, values);
db.close();
}
public void delete2(Integer id){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.delete("Student", "_id=?", new String[]{id.toString()});
}
public void delete(Integer id){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Delete from Student where _id=?",
new String[] { String.valueOf(id) });
db.close();
}
public void update2(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
ContentValues values=new ContentValues();
values.put("name", stu.getName());
values.put("sex",stu.getSex());
db.update("Student", values, "_id=?", new String[]{String.valueOf(stu.get_id())});
}
public void update(Student stu){
SQLiteDatabase db=dbhelper.getWritableDatabase();
db.execSQL("Update Student Set name=?,sex=? where _id=?",
new String[] {stu.getName(),stu.getSex(),String.valueOf(stu.get_id()) });
db.close();
}
public Student query(Integer _id){
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf=db.rawQuery("Select _id,name,sex from Student where _id=?",
new String[] {String.valueOf(_id)});
if(stuimf!=null){
if(stuimf.moveToFirst()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
return new Student(id,name,sex);
}
}
db.close();
return null;
}
public Student query2(Integer _id){
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf= db.query("Student", new String[]{"_id","name","sex"}, "_id=?",new String[]{_id.toString()}, null, null, null);
if(stuimf!=null){
if(stuimf.moveToFirst()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
return new Student(id,name,sex);
}
}
return null;
}
/** * 分页查询 * @param offset 跳过前面多少记录 * @param maxRecord 每页获取多少记录 * @return */
public List<Student> getScrollData(int offset,int maxRecord){
SQLiteDatabase db=dbhelper.getReadableDatabase();
List<Student> stulist = new ArrayList<Student>();
Cursor stuimf=db.rawQuery("Select _id,name,sex from Student order by _id asc limit ?,?",
new String[] {String.valueOf(offset),String.valueOf(maxRecord)});
if(stuimf!=null){
while(stuimf.moveToNext()){
int id=stuimf.getInt(stuimf.getColumnIndex("_id"));
String name=stuimf.getString(stuimf.getColumnIndex("name"));
String sex=stuimf.getString(stuimf.getColumnIndex("sex"));
stulist.add(new Student(id,name,sex));
}
return stulist;
}
db.close();
return null;
}
public long getCount(){
long result=0;
SQLiteDatabase db=dbhelper.getReadableDatabase();
Cursor stuimf=db.rawQuery("Select count(*) from Student", null);
if(stuimf!=null){
//因为只要创建了表,count(*)最小值为1,即还未插入数据时
stuimf.moveToFirst();
result=stuimf.getLong(0);
}
return result;
}
}
至此,完成了SQLite的基本应用的所有工作。我们再来回顾下几个对象的主要作用,SQLiteOpenHelper用于创建数据库、数据库结构和得到真正操作数据库的实例SQLiteDatabase;SQLiteDatabase直接完成数据的各种操作;Cursor对象用于把查询集合封装起来;而ContentValus也是封装数据,把数据列和对应的值封装成为键值对。欲了解更多可以参考电子书SQLite权威指南SQLite也是有很多图形界面工具的使用最多的就是SQLite Expert安装包