SQLite

以下内容整理自互联网,仅用于个人学习


1. Android保存数据

Android提供几种保存数据的方式,保证程序在结束之后这些保存的数据不会丢失。

  • 文本文件保存:可以保存在应用程序自己的目录下,安装的每个app都会在/data/data/目录下创建个文件夹,名字和应用程序中AndroidManifest.xml文件中的package一样。
  • SD卡保存
  • Preferences保存:这也是一种经常使用的数据存储方法,因为它们对于用户而言是透明的,并且从应用安装的时候就存在了。
  • Assets保存:用来存储一些只读数据,Assets是指那些在assets目录下的文件,这些文件在你将你的应用编译打包之前就要存在,并且可以在应用程序运行的时候被访问到。

但有时候我们需要对保存的数据进行一些复杂的操作,或者数据量很大,超出了文本文件和Preference的性能能的范围,所以需要一些更加高效的方法来管理,从Android1.5开始,Android就自带SQLite数据库了。

SQLite它是一个独立的,无需服务进程,支持事务处理,可以使用SQL语言的数据库。

2. SQLite特性

  • ACID事务 :指数据库事务正确执行的四个基本要素的缩写。包含:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。一个支持事务(Transaction)的数据库,必需要具有这四种特性,否则在事务过程(Transaction processing)当中无法保证数据的正确性,交易过程极可能达不到交易方的要求。
  • 无需安装和管理配置
  • 支持多种开发语言,C,PHP,Perl,Java,ASP.NET,Python
  • 储存在单一磁盘文件中的一个完整的数据库
  • 数据库文件可以在不同字节顺序的机器间自由的共享
  • 支持数据库大小至2TB
  • 在普通数据库操作方面比一些流行的数据库要快
  • 简单, 轻松的API
  • 包含TCL绑定, 同时通过Wrapper支持其他语言的绑定
  • 良好注释的源代码, 并且有着90%以上的测试覆盖率
  • 独立: 没有额外依赖

3. Android中使用SQLite

3.1 创建数据库

Android 不自动提供数据库。在 Android 应用程序中使用 SQLite,必须自己创建数据库,然后创建表、索引,填充数据。Android 提供了 SQLiteOpenHelper 帮助你创建一个数据库,你只要继承 SQLiteOpenHelper 类根据开发应用程序的需要,封装创建和更新数据库使用的逻辑就行了。

SQLiteOpenHelper是一个抽象类,在这个类里有两个抽象方法,OnCreate和OnUpgrade,前者用于第一次创建数据库,后者用于数据库升级。

publicclassDatabaseHelperextendsSQLiteOpenHelper{

/**
*@paramcontext上下文环境(例如,一个Activity)
*@paramname数据库名字
*@paramfactory一个可选的游标工厂(通常是Null)
*@paramversion数据库模型版本的整数
*
*会调用父类SQLiteOpenHelper的构造函数
*/
publicDatabaseHelper(Contextcontext,Stringname,CursorFactoryfactory,intversion){
super(context,name,factory,version);

}

/**
*在数据库第一次创建的时候会调用这个方法
*
*根据需要对传入的SQLiteDatabase对象填充表和初始化数据。
*/
@Override
publicvoidonCreate(SQLiteDatabasedb){

}

/**
*当数据库需要修改的时候(两个数据库版本不同),Android系统会主动的调用这个方法。
*一般我们在这个方法里边删除数据库表,并建立新的数据库表.
*/
@Override
publicvoidonUpgrade(SQLiteDatabasedb,intoldVersion,intnewVersion){
//三个参数,一个SQLiteDatabase对象,一个旧的版本号和一个新的版本号

}

@Override
publicvoidonOpen(SQLiteDatabasedb){
//每次成功打开数据库后首先被执行
super.onOpen(db);
}
}

当Android应用运行时,SQLiteOpenHelper会先检查是否已经存在数据库,如果不存在,就创建数据库,然后打开数据库,最后调用OnCreate方法;如果数据库已存在,而版本号比上次创建的数据库版本号高,就调用OnUpgrade,用于升级。

继承SQLiteOpenHelper之后就拥有了以下两个方法:

  • getReadableDatabase()  创建或者打开一个查询数据库
  • getWritableDatabase() 创建或者打开一个可写数据库
DatabaseHelperdatabase=newDatabaseHelper(context);//传入一个上下文参数
SQLiteDatabasedb=null;
db=database.getWritableDatabase();

上面这段代码会返回一个 SQLiteDatabase 类的实例,使用这个对象,你就可以查询或者修改数据库。

3.2 创建表和索引

创建表和索引,需要调用 SQLiteDatabase 的 execSQL() 方法来执行 DDL 语句。如:

db.execSQL("CREATETABLEuser(_idINTEGERPRIMARYKEY
AUTOINCREMENT,usernameTEXT,passwordTEXT);");

要删除表和索引,需要使用 execSQL() 方法调用 DROP INDEX 和 DROP TABLE 语句。

3.3添加数据

Android中SQLite数据操作(添加、删除、更新)方式都有两种,第一种是使用execSQL() 方法执行 INSERT, UPDATE, DELETE 等语句,第二种是使用SQLiteDatabase 对象的 insert()、update()、delete()等方法。

3.3.1 使用execSQL() 添加数据

execSQL() 方法适用于所有不返回结果的 SQL 语句。

Stringsql="insertintouser(username,password)values('afsds','123456');//插入操作的SQL语句
db.execSQL(sql);//执行SQL语句

3.3.2 使用 SQLiteDatabase 对象的 insert()

ContentValuesvalue=newContentValues();
value .put("username","finch");//添加用户名
value .put("password","123456");//添加密码
db.insert("user",null,value );//执行插入操作

3.4 更新数据

3.4.1 使用execSQL方式的实现

Stringsql="update[user]setpassword='654321'whereusername="finch";//修改的SQL语句
db.execSQL(sql);//执行修改

3.4.2 使用SQLiteDatabase 对象的 update()方法

ContentValuesvalue=newContentValues();
value.put("password","654321");//添加要更改的字段及内容
StringwhereClause="username=?";//修改条件
String[]whereArgs={"finch"};//修改条件的参数
db.update("user",value,whereClause,whereArgs);//执行修改

该方法有四个参数:表名;列名和值的 ContentValues 对象;可选的 WHERE 条件;可选的填充 WHERE 语句的字符串,这些字符串会替换 WHERE 条件中的“?”标记,update() 根据条件,更新指定列的值。

3.5 删除数据

3.5.1 使用execSQL方式的实现

Stringsql="deletefromuserwhereusername="finch";//删除操作的SQL语句
db.execSQL(sql);//执行删除操作

3.5.2 使用SQLiteDatabase 对象的delete()方法

StringwhereClause="username=?";//删除的条件
String[]whereArgs={"finch"};//删除的条件参数
db.delete("user",whereClause,whereArgs);//执行删除

3.6 查询数据

查询数据不同于添加、删除、更新。主要有以下几种方式。

3.6.1 通过query实现查询

query() 方法用 SELECT 语句段构建查询。SELECT 语句内容作为 query() 方法的参数,比如:要查询的表名,要获取的字段名,WHERE 条件,包含可选的位置参数,去替代 WHERE 条件中位置参数的值,GROUP BY 条件,HAVING 条件。除了表名,其他参数可以是 null。

Cursorc=db.query("user",null,null,null,null,null,null);//查询并获得游标
if(c.moveToFirst()){//判断游标是否为空
for(inti=0;i<c.getCount();i++){ 
      c.move(i);//移动到指定记录
      Stringusername=c.getString(c.getColumnIndex("username");
      Stringpassword=c.getString(c.getColumnIndex("password"));
}
}

3.6.2 使用 rawQuery() 直接调用 SELECT 语句

Cursorc=db.rawQuery("select*fromuserwhereusername=?",newStirng[]{"finch"});

if(c.moveToFirst()){
Stringpassword=c.getString(c.getColumnIndex("password"));
}

返回值是一个 cursor 对象,这个对象的方法可以迭代查询结果。 如果查询是动态的,使用这个方法就会非常复杂。例如,当你需要查询的列在程序编译的时候不能确定,这时候使用 query() 方法会方便很多。

3.6.3 使用游标

不管你如何执行查询,都会返回一个 Cursor,这是 Android 的 SQLite 数据库游标。

  • 通过使用 getCount() 方法得到结果集中有多少记录;
  • 通过 moveToFirst(), moveToNext(), 和 isAfterLast() 方法遍历所有记录;
  • 通过 getColumnNames() 得到字段名;
  • 通过 getColumnIndex() 转换成字段号;
  • 通过 getString(),getInt() 等方法得到给定字段当前记录的值;
  • 通过 requery() 方法重新执行查询得到游标;
  • 通过 close() 方法释放游标资源;

下面给出一个遍历的示例

Cursorresult = db.rawQuery("SELECT_id,username,passwordFROMuser");
result.moveToFirst();
while(!result.isAfterLast()){
intid=result.getInt(0);
Stringname=result.getString(1);
Stringpassword=result.getString(2);
//dosomethingusefulwiththese
result.moveToNext();
 }
result.close();
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,009评论 25 707
  • 一.SQLite的介绍1.SQLite简介SQLite是一款轻型的数据库,是遵守ACID的关联式数据库管理系统,它...
    AiPuff阅读 616评论 0 3
  • SQLite数据库,和其他的SQL数据库不同, 我们并不需要在手机上另外安装一个数据库软件,Android系统已经...
    JuSong阅读 1,919评论 0 3
  • html转为NSAttributedString NSAttributedString 转为 html(作为NSA...
    SSBun阅读 3,247评论 1 4
  • 三国乱世之中似乎只有横扫千军的勇武和十步杀一人的谋略才被奉为行天下无阻的王道,大多数的歌功颂德不过是孱弱文人们迫于...
    闻心ol阅读 449评论 0 5