iOS中一些时间比较长的操作都应该放在子线程中,以避免UI的卡顿。而sqlite 是非线程安全的,故在多线程中不能共用同一个数据库连接,否则会导致EXC_BAD_ACCESS。所以我们可以在子线程中创建一个新的db连接(新建一个db对象),然后再操作数据库。

如果选择FMDB,除了以上所说的方式外,还可以利用FMDatabaseQueue来解决多线程问题。

关于FMDB的使用网上教程比较多,内容基本都是与唐巧的这篇http://www.devtang.com/blog/2012/04/22/use-fmdb/差不多

这里记录一下多线程下的使用。

方式一:采用每次新建db的方式

db路径我是写了一个方法

+ (NSString *)getDBPath
{
NSString* docsdir = [NSSearchPathForDirectoriesInDomains( NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
NSFileManager *filemanage = [NSFileManager defaultManager];
docsdir = [docsdir stringByAppendingPathComponent:@"FMDBDemo"];
BOOL isDir;
BOOL exit =[filemanage fileExistsAtPath:docsdir isDirectory:&isDir];
if (!exit || !isDir) {
[filemanage createDirectoryAtPath:docsdir withIntermediateDirectories:YES attributes:nil error:nil];
}
NSString *dbpath = [docsdir stringByAppendingPathComponent:@"myDB.sqlite"];
return dbpath;
}

1、创建数据库表

/** 创建表 */
+ (BOOL)createUserTableByDB
{
FMDatabase *db = [FMDatabase databaseWithPath:[self getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} NSString *sql = @"CREATE TABLE IF NOT EXISTS User(ID INTEGER PRIMARY KEY, name TEXT, age INTEGER, ID_No TEXT);";
BOOL res = [db executeUpdate:sql];
if (res) {
NSLog(@"创建表格成功");
}else {
NSLog(@"创建表格失败");
}
return res;
}

2、插入数据

插入数据,我分了两种情形,一种是单条插入,一种是利用事务(利用事务插入多条时,时间会快很多,而且有任何一条插入失败,则不会提交到数据库)

/** 保存单个对象 */
- (BOOL)saveByDB
{
FMDatabase *db = [FMDatabase databaseWithPath:[User getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
}
NSString *sql = [NSString stringWithFormat:@"INSERT INTO User(name, age, ID_No) VALUES ('%@', '%d', '%@');", self.name, self.age, self.ID_no];
BOOL res = [db executeUpdate:sql];
[db close];
if (res) {
NSLog(@"插入数据成功");
}else {
NSLog(@"插入数据失败");
}
return res;
} /** 批量保存用户对象 */
+ (BOOL)saveObjectsByDB:(NSArray *)array
{
FMDatabase *db = [FMDatabase databaseWithPath:[self getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} [db beginTransaction]; BOOL isRollBack = NO;
@try {
for (User *user in array) {
NSString *sql = [NSString stringWithFormat:@"INSERT INTO User(name, age, ID_No) VALUES ('%@', '%d', '%@');", user.name, user.age, user.ID_no];
BOOL res = [db executeUpdate:sql];
if (!res) {
NSLog(@"db事务插入失败");
}else {
NSLog(@"db事务插入成功");
}
}
}
@catch (NSException *exception) {
isRollBack = YES;
[db rollback];
}
@finally {
if (!isRollBack) {
[db commit];
}
}
[db close]; return !isRollBack;
}

更新与删除也分两种情形,与插入类似。

3、更新数据

/** 更新单个对象 */
- (BOOL)updateByDB
{
FMDatabase *db = [FMDatabase databaseWithPath:[User getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} NSString *sql = [NSString stringWithFormat:@"UPDATE User SET name = '%@',age = '%d',ID_no = '%@' WHERE ID = '%d';", self.name, self.age, self.ID_no, self.ID];
BOOL res = [db executeUpdate:sql];
[db close];
if (res) {
NSLog(@"插入数据成功");
}else {
NSLog(@"插入数据失败");
} return res;
} /** 批量更新用户对象*/
+ (BOOL)updateObjectsByDB:(NSArray *)array
{
FMDatabase *db = [FMDatabase databaseWithPath:[User getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} [db beginTransaction]; BOOL isRollBack = NO;
@try {
for (User *user in array) {
NSString *sql = [NSString stringWithFormat:@"UPDATE User SET name = '%@',age = '%d',ID_no = '%@' WHERE ID = '%d';", user.name, user.age, user.ID_no, user.ID];
BOOL res = [db executeUpdate:sql];
if (!res) {
NSLog(@"db事务更新失败");
}else {
NSLog(@"db事务更新成功");
}
}
}
@catch (NSException *exception) {
isRollBack = YES;
[db rollback];
}
@finally {
if (!isRollBack) {
[db commit];
}
}
[db close];
return !isRollBack;
}

4、删除数据

/** 删除单个对象 */
- (BOOL)deleteObjectByDB
{
FMDatabase *db = [FMDatabase databaseWithPath:[User getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} NSString *sql = [NSString stringWithFormat:@"DELETE FROM User WHERE ID = '%d'",self.ID];
BOOL res = [db executeUpdate:sql];
[db close];
if (res) {
NSLog(@"删除数据成功");
}else {
NSLog(@"删除数据失败");
}
return res;
} /** 批量删除用户对象 */
+ (BOOL)deleteObjectsByDB:(NSArray *)array
{
FMDatabase *db = [FMDatabase databaseWithPath:[self getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return NO;
} [db beginTransaction]; BOOL isRollBack = NO;
@try {
for (User *user in array) {
NSString *sql = [NSString stringWithFormat:@"DELETE FROM User WHERE ID = '%d'",user.ID];
BOOL res = [db executeUpdate:sql];
if (!res) {
NSLog(@"db事务删除失败");
}else {
NSLog(@"db事务删除成功");
}
}
}
@catch (NSException *exception) {
isRollBack = YES;
[db rollback];
}
@finally {
if (!isRollBack) {
[db commit];
}
}
[db close];
return !isRollBack;
}

5、查询数据

查询就比较简单了。

/** 查询全部用户 */
+ (NSArray *)findAllByDB
{
FMDatabase *db = [FMDatabase databaseWithPath:[self getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return nil;
} NSMutableArray *users = [[NSMutableArray alloc] init];
NSString *sql = @"SELECT * FROM User";
FMResultSet *resultSet = [db executeQuery:sql];
while ([resultSet next]) {
User *user = [[User alloc] init];
user.ID = [resultSet intForColumn:@"ID"];
user.name = [resultSet stringForColumn:@"name"];
user.age = [resultSet intForColumn:@"age"];
user.ID_no = [resultSet stringForColumn:@"ID_no"];
[users addObject:user];
[user release];
}
NSLog(@"查询全部成功");
return [users autorelease];
} /** 查找某个用户 */
+ (instancetype)findFirstBySqlByDB:(NSString *)sql
{
User *user = nil;
NSArray *users = [self findBySqlByDB:sql];
if (users.count > 0) {
user = [users firstObject];
}
return user;
} /** 查找用户 */
+ (NSArray *)findBySqlByDB:(NSString *)sql
{
FMDatabase *db = [FMDatabase databaseWithPath:[self getDBPath]];
if (![db open]) {
NSLog(@"数据库打开失败!");
return nil;
} NSMutableArray *users = [[NSMutableArray alloc] init];
NSString *select = @"SELECT * FROM User ";
if (sql) {
select = [select stringByAppendingString:sql];
}
FMResultSet *resultSet = [db executeQuery:select];
while ([resultSet next]) {
User *user = [[User alloc] init];
user.ID = [resultSet intForColumn:@"ID"];
user.name = [resultSet stringForColumn:@"name"];
user.age = [resultSet intForColumn:@"age"];
user.ID_no = [resultSet stringForColumn:@"ID_no"];
[users addObject:user];
[user release];
}
NSLog(@"条件查询成功");
return [users autorelease];
}

而多线程下的调用方式是这样的,其中上面的所有方法都是在对象类User中。

/** 用db插入User数据*/
- (IBAction)dbInsertData:(id)sender { //多线程插入数据
for (int i = 0; i < 5; i++) {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
User *user = [[User alloc] init];
user.name = @"dbName一";
user.ID_no = [NSString stringWithFormat:@"%d",55555+i];
user.age = 555+i;
[user saveByDB];
});
} //利用事务插入数据
dispatch_async(dispatch_get_global_queue(0, 0), ^{
NSMutableArray *array = [NSMutableArray array];
for (int i = 0; i < 5; i++) {
User *user = [[User alloc] init];
user.name = @"db事务";
user.ID_no = [NSString stringWithFormat:@"%d",66666+i];
user.age = 66+i;
[array addObject:user];
[user release];
} [User saveObjectsByDB:array];
});
}

iOS下FMDB的多线程操作(一)的更多相关文章

  1. iOS下FMDB的多线程操作(二)

    上一篇记录不使用FMDatabaseQueue来使用多线程,这一篇记录一下使用FMDatabaseQueue的方式. 需要注意的时queue操作中不能嵌套queue操作,否则会各种错误. 当使用FMD ...

  2. iOS——使用FMDB进行数据库操作(转载)

    iOS 使用FMDB进行数据库操作 https://github.com/ccgus/fmdb [摘要]本文介绍iOS 使用FMDB进行数据库操作,并提供详细的示例代码供参考. FMDB 使用方法 A ...

  3. swift语言之多线程操作和操作队列(下)———坚持51天吃掉大象(写技术文章)

    欢迎有兴趣的朋友,参与我的美女同事发起的活动<51天吃掉大象>,该美女真的很疯狂,希望和大家一起坚持51天做一件事情,我加入这个队伍,希望坚持51天每天写一篇技术文章.关注她的微信公众号: ...

  4. Ios第三方FMDB使用说明

    SQLite (http://www.sqlite.org/docs.html) 是一个轻量级的关系数据库.iOS SDK很早就支持了SQLite,在使用时,只需要加入 libsqlite3.dyli ...

  5. iOS进阶面试题----多线程

    1 多线程是什么 多线程是个复杂的概念,按字面意思是同步完成多 项任务,提高了资源的使用效率,从硬件.操作系统.应用软件不同的角度去看,多线程被赋予不同的内涵,对于硬件,现在市面上多数的CPU都是多核 ...

  6. iOS中FMDB的使用

    1在日常的开发中,我们需要用到离线缓存将数据信息存入数据库,在没有网络的时候进行加载,而我们IOS用的就是sqlite3数据库,用原生的sql我们也能实现,但是书写起来比较麻烦,尤其是其它语言转过来的 ...

  7. iOS的三种多线程技术NSThread/NSOperation/GCD

    1.iOS的三种多线程技术 1.NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程) 2.以下两点是苹果专门开发的"并发"技术,使得程序员可以不再去关心 ...

  8. 一行代码实现FMDB的CURD操作

    上次实现FMDB的CURD基本操作后,用在项目里,每个实体类都要写SQL语句来实现创建表和CURD操作,总觉得太麻烦,然后就想着利用反射和kvc来实现一个数据库操作的基类继承一下,子类只需要继承,然后 ...

  9. IOS高级开发之多线程(四)NSOperation

    1.什么是NSOperation,NSOperationQueue? NSOperation是一个抽象的基类,表示一个独立的计算单元,可以为子类提供有用且线程安全的建立状态,优先级,依赖和取消等操作. ...

随机推荐

  1. DOM4J生成、解析XML实例

    import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.util.Iterato ...

  2. FORM级别和数据库级别的Trace

     metalink上的文章较全的,中文的可参考我的博客EBS开发技术之trace http://blog.csdn.net/cai_xingyun/article/details/17250971 ...

  3. SSH网上商城---使用ajax完成用户名是否存在异步校验

    小伙伴在上网的时候,需要下载或者观看某些视频资料,更或者是在逛淘宝的时候,我们都需要注册一个用户,当我们填写好各种信息,点击确定的时候,提示用户名已经存在,小编就想,为什么当我们填写完用户名的时候,她 ...

  4. 贪婪算法(Greedy algorithm)-算法学习之旅(一)

    我研一下学期选修了网络管理,因为是限选课所以我没有认真上课,今天因为快结课考试了,我才去,然后我发现了一个大秘密...... 原来老师从第三节课已经开始讲算法导论的知识了,我是不是错过了什么... 废 ...

  5. 一步步创建Qt Widget项目+TextFinder案例(摘自笔者2015年将出的《QT5权威指南》,本文为试读篇)

     创建一个基于应用的QtWidget应用程序 这个手册描述了怎样使用QtCreater创建个一个小的Qt应用程序,Text Finder.它是Qt工具Text Finder例子的简写版本.这个应用 ...

  6. C++中所有的变量和函数都必须有类型

    /* C++中所有的变量和函数都必须有类型 C语言中的默认类型在C++中是不合法的 函数f的返回值是什么类型,参数又是什么类型? 函数g可以接受多少个参数? */ //更换成.cpp就会报错 f(i) ...

  7. 【Unity Shaders】游戏性和画面特效——创建一个夜视效果的画面特效

    本系列主要参考<Unity Shaders and Effects Cookbook>一书(感谢原书作者),同时会加上一点个人理解或拓展. 这里是本书所有的插图.这里是本书所需的代码和资源 ...

  8. (八十八)VFL语言初步 - 实现布局

    [基本语法] VFL的语法为H:和V:开头,代表水平和垂直. 接下来如果要涉及距离,使用|-x-,x为距离的点数. 对于视图,用[ ]包围,例如[blueView]. ①下面的语句实现了blueVie ...

  9. UNIX网络编程——套接字选项(SO_REUSEADDR)

    1.一般来说,一个端口释放后会等待两分钟之后才能再被使用,SO_REUSEADDR是让端口释放后立即就可以被再次使用. SO_REUSEADDR用于对TCP套接字处于TIME_WAIT状态下的sock ...

  10. HDFS追本溯源:HDFS操作的逻辑流程与源码解析

    本文主要介绍5个典型的HDFS流程,这些流程充分体现了HDFS实体间IPC接口和stream接口之间的配合. 1. Client和NN Client到NN有大量的元数据操作,比如修改文件名,在给定目录 ...