iOS数据持久化-SQLite数据库使用详解
使用SQLite数据库
创建数据库
创建数据库过程需要3个步骤:
1、使用sqlite3_open函数打开数据库;
2、使用sqlite3_exec函数执行Create Table语句,创建数据库表;
3、使用sqlite3_close函数释放资源。
这个过程中使用了3个SQLite3函数,它们都是纯C语言函数,通过Objective-C去调用C函数当然不是什么问题,但是也要注意Objective-C数据类型与C数据类型兼容性问题。
下面我们使用SQLite技术实现备忘录案例,与属性列表文件实现一样,我们只需要修改持久层工程(PersistenceLayer)中NoteDAO类就可以了。首先我们需要添加SQLite3库到工程环境中,有3个工程需要添加到哪个呢?应该添加到可以运行的工程即表示层工程PresentationLayer。选择工程PresentationLayer中TARGETS→PresentationLayer→Link Binary With Libraries,点击左下角的“+”,弹出对话框选择libsqlite3.dylib或libsqlite3.0.dylib,在弹出的对话框中点击Add添加。
NoteDAO.h文件的修改:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | # import ”Note.h” # import ”sqlite3.h” #define DBFILE_NAME @”NotesList.sqlite3″ @ interface NoteDAO : NSObject { sqlite3 *db; } + (NoteDAO*)sharedManager; - (NSString *)applicationDocumentsDirectoryFile; - ( void )createEditableCopyOfDatabaseIfNeeded; //插入Note方法 -( int ) create:(Note*)model; //删除Note方法 -( int ) remove:(Note*)model; //修改Note方法 -( int ) modify:(Note*)model; //查询所有数据方法 -(NSMutableArray*) findAll; //按照主键查询数据方法 -(Note*) findById:(Note*)model; @end |
我们需要使用语句#import ”sqlite3.h”引入sqlite3头文件,而且需要定义sqlite3*成员变量db。NoteDAO.m中的createEditableCopyOfDatabaseIfNeeded方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | - ( void )createEditableCopyOfDatabaseIfNeeded { NSString *writableDBPath = [self applicationDocumentsDirectoryFile]; if (sqlite3_open([writableDBPath UTF8String], &db) != SQLITE_OK) { ① sqlite3_close(db); ② NSAssert(NO,@”数据库打开失败。”); } else { char *err; NSString *createSQL = [NSString stringWithFormat:@"CREATE TABLE IF NOT EXISTS Note (cdate TEXT PRIMARY KEY, content TEXT);"]; ③ if (sqlite3_exec(db,[createSQL UTF8String],NULL,NULL,&err) != SQLITE_OK) { ④ sqlite3_close(db); ⑤ NSAssert1(NO, @”建表失败, %s”, err); ⑥ } sqlite3_close(db); ⑦ } } |
createEditableCopyOfDatabaseIfNeeded方法用于创建数据库,第1步打开数据库,代码①行,语句是sqlite3_open([writableDBPath UTF8String], &db),sqlite3_open函数的第1个参数是数据库文件完整的路径,但是需要注意的是在SQLite3函数中接受的是char*的UTF-8类型数据,需要将NSString*转换为UTF-8,使用NSString*的UTF8String方法可以转换,sqlite3_open函数第2个参数sqlite3指针变量db的地址。该函数的返回值是int类型,在SQLite3中定义了很多常量,返回值等于常量SQLITE_OK则说明操作成功。
第2步执行建表语句,代码第④行,语句sqlite3_exec(db,[createSQL UTF8String],NULL,NULL,&err)执行建表的SQL。第1个参数是sqlite3指针变量db的地址,第2个参数是要执行的sql语句,第3个参数是要回调函数,第4个参数是要回调函数的参数,第5个参数是执行出错的字符串。建表SQL语句是,如果表Note存在这不用创建。
CREATE TABLE IF NOT EXISTS Note (cdate TEXT PRIMARY KEY, content TEXT)
第3步使用sqlite3_close函数释放资源,代码②、⑤、⑦行所示,在数据库打开失败、Create Table执行失败和成功执行完成时候调用。原则上无论正常结束还是异常结束必须使用sqlite3_close函数释放资源。
查询数据
数据查询一般会带有查询条件,这个使用SQL语句where子句很容易实现,但是在程序中需要动态绑定参数给where子句。执行查询数据步骤如下:
1、使用sqlite3_open函数打开数据库;
2、使用sqlite3_prepare_v2函数预处理SQL语句;
3、使用sqlite3_bind_text函数绑定参数;
4、使用sqlite3_step函数执行SQL语句,遍历结果集;
5、使用sqlite3_column_text等函数提取字段数据;
6、使用sqlite3_finalize和sqlite3_close函数释放资源。
NoteDAO.m中的按照主键查询数据方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | -(Note*) findById:(Note*)model { NSString *path = [self applicationDocumentsDirectoryFile]; if (sqlite3_open([path UTF8String], &db) != SQLITE_OK) { ① sqlite3_close(db); ② NSAssert(NO,@”数据库打开失败。”); } else { NSString *qsql = @”SELECT cdate,content FROM Note where cdate =?”; sqlite3_stmt *statement; //预处理过程 if (sqlite3_prepare_v2(db, [qsql UTF8String], - 1 , &statement, NULL) == SQLITE_OK) { ③ //准备参数 NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; ④ [dateFormatter setDateFormat:@ "yyyy-MM-dd HH:mm:ss" ]; NSString *nsdate = [dateFormatter stringFromDate:model.date]; //绑定参数开始 sqlite3_bind_text(statement, 1 , [nsdate UTF8String], - 1 , NULL); ⑤ //执行 if (sqlite3_step(statement) == SQLITE_ROW) { ⑥ char *cdate = (char *) sqlite3_column_text(statement, 0 ); ⑦ NSString *nscdate = [[NSString alloc] initWithUTF8String: cdate]; char *content = (char *) sqlite3_column_text(statement, 1 ); NSString * nscontent = [[NSString alloc] initWithUTF8String: content]; Note* note = [[Note alloc] init]; note.date = [dateFormatter dateFromString:nscdate]; note.content = nscontent; sqlite3_finalize(statement); sqlite3_close(db); return note; } } sqlite3_finalize(statement); ⑧ sqlite3_close(db); ⑨ } return nil; } |
该方法执行了6个步骤,其中第1个步骤,代码第①行所示,它与创建数库的第1个步骤是一样的,不用再介绍了。
第2个步骤,代码第③行所示,语句sqlite3_prepare_v2(db, [qsql UTF8String], -1, &statement, NULL)是预处理SQL语句,预处理目的是将SQL编译成二进制代码,提高SQL语句执行的速度。sqlite3_prepare_v2函数的第3个参数-1代表全部sql字符串长度,第4个参数&statement是sqlite3_stmt指针的地址,它是语句对象,通过语句对象可以执行SQL语句,第5个参数是sql语句没有被执行的部分语句。
第3个步骤,代码第⑤行所示,语句sqlite3_bind_text(statement, 1, [nsdate UTF8String], -1, NULL)是绑定SQL语句参数。在SQL语句中带有问号,这个问号就是要绑定的参数,问号是占位符。
NSString *qsql = @”SELECT cdate,content FROM Note where cdate =?”;
sqlite3_bind_text函数是绑定参数,第1个参数是statement指针,第2个参数为序号(从1开始),第3个参数为字符串值,第4个参数为字符串长度,第5个参数为一个函数指针。
第4个步骤sqlite3_step(statement)执行SQL语句,代码第⑥行所示,sqlite3_step返回int类型,等于SQLITE_ROW说明还要其它的行没有遍历。
第5个步骤提取字段数据,代码第⑦行所示,使用sqlite3_column_text(statement, 0)函数可以读取字符串类型字段,第2参数是指定select字段的索引(从0开始)。同样char*转换成为NSString*类型,需要initWithUTF8String:构造方法。读取字段函数采用与字段类型有关系,SQLite3的类似的常用函数还有:
sqlite3_column_blob()
sqlite3_column_double()
sqlite3_column_int()
sqlite3_column_int64()
sqlite3_column_text()
sqlite3_column_text16()
关于其它的API可以参考http://www.sqlite.org/cintro.html。
第6个步骤是释放资源,创建数据库过程不同,除了使用sqlite3_close函数关闭数据库,代码第⑧行所示,还要使用sqlite3_finalize函数释放语句对象statement代码第⑨行所示。
NoteDAO.m中的查询所有数据方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | -(NSMutableArray*) findAll { NSString *path = [self applicationDocumentsDirectoryFile]; NSMutableArray *listData = [[NSMutableArray alloc] init]; if (sqlite3_open([path UTF8String], &db) != SQLITE_OK) { sqlite3_close(db); NSAssert(NO,@”数据库打开失败。”); } else { NSString *qsql = @”SELECT cdate,content FROM Note”; sqlite3_stmt *statement; //预处理过程 if (sqlite3_prepare_v2(db, [qsql UTF8String], - 1 , &statement, NULL) == SQLITE_OK) { NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; [dateFormatter setDateFormat:@ "yyyy-MM-dd HH:mm:ss" ]; //执行 while (sqlite3_step(statement) == SQLITE_ROW) { char *cdate = (char *) sqlite3_column_text(statement, 0 ); NSString *nscdate = [[NSString alloc] initWithUTF8String: cdate]; char *content = (char *) sqlite3_column_text(statement, 1 ); NSString * nscontent = [[NSString alloc] initWithUTF8String: content]; Note* note = [[Note alloc] init]; note.date = [dateFormatter dateFromString:nscdate]; note.content = nscontent; [listData addObject:note]; } } sqlite3_finalize(statement); sqlite3_close(db); } return listData; } |
查询所有数据方法与按照主键查询数据方法类似,区别在于本方法没有查询条件不需要绑定参数。遍历的时候使用while循环语句,不是if语句。
while (sqlite3_step(statement) == SQLITE_ROW) {
… …
}
修改数据
修改数据包括:insert、update和delete语句。这3个SQL语句都可以带有参数,关于参数的绑定与查询where子句绑定的方式是一样的。执行修改数据步骤如下:
1、使用sqlite3_open函数打开数据库;
2、使用sqlite3_prepare_v2函数预处理SQL语句;
3、使用sqlite3_bind_text函数绑定参数;
4、使用sqlite3_step函数执行SQL语句;
5、使用sqlite3_finalize和sqlite3_close函数释放资源。
修改数据的步骤与查询数据的步骤相比少了一个提取字段数据步骤。下面我们看看代码部分。其它的步骤是一样的。
NoteDAO.m中的插入Note方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 | -( int ) create:(Note*)model { NSString *path = [self applicationDocumentsDirectoryFile]; if (sqlite3_open([path UTF8String], &db) != SQLITE_OK) { ① sqlite3_close(db); ② NSAssert(NO,@”数据库打开失败。”); } else { NSString *sqlStr = @”INSERT OR REPLACE INTO note (cdate, content) VALUES (?,?)”; sqlite3_stmt *statement; //预处理过程 if (sqlite3_prepare_v2(db, [sqlStr UTF8String], - 1 , &statement, NULL) == SQLITE_OK) { ③ NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; [dateFormatter setDateFormat:@ "yyyy-MM-dd HH:mm:ss" ]; NSString *nsdate = [dateFormatter stringFromDate:model.date]; //绑定参数开始 sqlite3_bind_text(statement, 1 , [nsdate UTF8String], - 1 , NULL); ④ sqlite3_bind_text(statement, 2 , [model.content UTF8String], - 1 , NULL); //执行插入 if (sqlite3_step(statement) != SQLITE_DONE) { ⑤ NSAssert(NO, @”插入数据失败。”); } } sqlite3_finalize(statement); ⑥ sqlite3_close(db); ⑦ } return 0 ; } |
第⑤行代码sqlite3_step(statement)语句执行插入语句,常量SQLITE_DONE执行完成。
NoteDAO.m中的删除Note方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 | -( int ) remove:(Note*)model { NSString *path = [self applicationDocumentsDirectoryFile]; if (sqlite3_open([path UTF8String], &db) != SQLITE_OK) { sqlite3_close(db); NSAssert(NO,@”数据库打开失败。”); } else { NSString *sqlStr = @”DELETE from note where cdate =?”; sqlite3_stmt *statement; //预处理过程 if (sqlite3_prepare_v2(db, [sqlStr UTF8String], - 1 , &statement, NULL) == SQLITE_OK) { NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; [dateFormatter setDateFormat:@ "yyyy-MM-dd HH:mm:ss" ]; NSString *nsdate = [dateFormatter stringFromDate:model.date]; //绑定参数开始 sqlite3_bind_text(statement, 1 , [nsdate UTF8String], - 1 , NULL); //执行插入 if (sqlite3_step(statement) != SQLITE_DONE) { NSAssert(NO, @”删除数据失败。”); } } sqlite3_finalize(statement); sqlite3_close(db); } return 0 ; } NoteDAO.m中的修改Note方法: -( int ) modify:(Note*)model { NSString *path = [self applicationDocumentsDirectoryFile]; if (sqlite3_open([path UTF8String], &db) != SQLITE_OK) { sqlite3_close(db); NSAssert(NO,@”数据库打开失败。”); } else { NSString *sqlStr = @”UPDATE note set content=? where cdate =?”; sqlite3_stmt *statement; //预处理过程 if (sqlite3_prepare_v2(db, [sqlStr UTF8String], - 1 , &statement, NULL) == SQLITE_OK) { NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; [dateFormatter setDateFormat:@ "yyyy-MM-dd HH:mm:ss" ]; NSString *nsdate = [dateFormatter stringFromDate:model.date]; //绑定参数开始 sqlite3_bind_text(statement, 1 , [model.content UTF8String], - 1 , NULL); sqlite3_bind_text(statement, 2 , [nsdate UTF8String], - 1 , NULL); //执行插入 if (sqlite3_step(statement) != SQLITE_DONE) { NSAssert(NO, @”修改数据失败。”); } } sqlite3_finalize(statement); sqlite3_close(db); } return 0 ; } |

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
cocos2d-x 创建工程详解
我们的编写的第一个程序一般习惯上都命名为HelloWorld,从它开始再学习其他的内容。下面介绍的第一个Cocos2d-x游戏我们也命名为HelloWorld。 创建工程 在Cocos2d-x早期版本中,创建工程是通过安装在Visual Studio中的工程模板而创建的,而目前创建工程Cocos2d-x是通过Cocos2d-x提供的命令工具cocos实现的,cocos位于<Cocos2d-x安装目录>\tools\cocos2d-console\bin下。我们能够通过DOS等终端进入bin目录执行下边的指令: cocos new HelloWorld-p com.work6 -l cpp -d D:/projects 其中D:/projects为HelloWorld的工程生成目录。通过上面的指令我们在D:/projects目录下面生成了名为HelloWorld的Cocos2d-x工程。打开HelloWorld目录。 从图中可以看出生成的工程代码是适合于多平台的,其中Classes目录是放置一些通用类(与平台无关的),我们编写的C++代码主要放置在该目录下面。图中cocos2...
- 下一篇
即时通讯二次开发源码
用于二次开发企业即时通讯源码(C++/MFC),现在,做一个普通的即时通讯软件不是一件难事,网上有很多很好的这类软件的源代码可以拿来使用。今天就介绍一个我用过,觉得不错的给大家。软件叫 ANYCHAT SDK(以下简称SDK),要做一个 QQ 一样的即时通讯,很难,也没那个必要。我用这个 SDK的目的只有一个,就是供我们企业使用。 SDK本身只有非常简单的即时通讯功能,它最大的优点就是简单稳定、并发数巨大,我拿到之后,做了一些针对我们企业内部特殊需求的修改,最后我们公司的这个SDK集成了企业的 ERP、OA、工作管理等软件。大家工作效率很明显提高了。 SDK是一款C/S模式的企业即时通讯系统,适合企业使用。 企业员工可在企业内部或外部通过SDK进行通讯,支持消息发送,文件传输,语音视频对话等。 统一的用户管理,通讯记录保存在主机或服务器,查询方便。 为企业提供安全,稳定的即时通讯解决方案。 开发语言采用 C++: VC++ 6.0、 Visual Studio系列(VS2003、VS2005、VS2008以及最新的VS2010) .Net: VB.Net、C#、WPF Delphi: ...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- CentOS8编译安装MySQL8.0.19
- CentOS关闭SELinux安全模块
- CentOS6,CentOS7官方镜像安装Oracle11G
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS7,8上快速安装Gitea,搭建Git服务器