iOS下FMDB的多线程操作(一)
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的多线程操作(一)的更多相关文章
- iOS下FMDB的多线程操作(二)
上一篇记录不使用FMDatabaseQueue来使用多线程,这一篇记录一下使用FMDatabaseQueue的方式. 需要注意的时queue操作中不能嵌套queue操作,否则会各种错误. 当使用FMD ...
- iOS——使用FMDB进行数据库操作(转载)
iOS 使用FMDB进行数据库操作 https://github.com/ccgus/fmdb [摘要]本文介绍iOS 使用FMDB进行数据库操作,并提供详细的示例代码供参考. FMDB 使用方法 A ...
- swift语言之多线程操作和操作队列(下)———坚持51天吃掉大象(写技术文章)
欢迎有兴趣的朋友,参与我的美女同事发起的活动<51天吃掉大象>,该美女真的很疯狂,希望和大家一起坚持51天做一件事情,我加入这个队伍,希望坚持51天每天写一篇技术文章.关注她的微信公众号: ...
- Ios第三方FMDB使用说明
SQLite (http://www.sqlite.org/docs.html) 是一个轻量级的关系数据库.iOS SDK很早就支持了SQLite,在使用时,只需要加入 libsqlite3.dyli ...
- iOS进阶面试题----多线程
1 多线程是什么 多线程是个复杂的概念,按字面意思是同步完成多 项任务,提高了资源的使用效率,从硬件.操作系统.应用软件不同的角度去看,多线程被赋予不同的内涵,对于硬件,现在市面上多数的CPU都是多核 ...
- iOS中FMDB的使用
1在日常的开发中,我们需要用到离线缓存将数据信息存入数据库,在没有网络的时候进行加载,而我们IOS用的就是sqlite3数据库,用原生的sql我们也能实现,但是书写起来比较麻烦,尤其是其它语言转过来的 ...
- iOS的三种多线程技术NSThread/NSOperation/GCD
1.iOS的三种多线程技术 1.NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程) 2.以下两点是苹果专门开发的"并发"技术,使得程序员可以不再去关心 ...
- 一行代码实现FMDB的CURD操作
上次实现FMDB的CURD基本操作后,用在项目里,每个实体类都要写SQL语句来实现创建表和CURD操作,总觉得太麻烦,然后就想着利用反射和kvc来实现一个数据库操作的基类继承一下,子类只需要继承,然后 ...
- IOS高级开发之多线程(四)NSOperation
1.什么是NSOperation,NSOperationQueue? NSOperation是一个抽象的基类,表示一个独立的计算单元,可以为子类提供有用且线程安全的建立状态,优先级,依赖和取消等操作. ...
随机推荐
- ssh用法及命令
http://blog.csdn.net/pipisorry/article/details/52269785 什么是SSH? 简单说,SSH是一种网络协议,用于计算机之间的加密登录.如果一个用户从本 ...
- Java命名和目录接口——JNDI
JNDI即Java命名和目录接口(JavaNaming and Directory Interface),它属于J2EE规范范畴,是J2EE的核心技术之一,提供了一组接口.类和关于命名空间的概念.JD ...
- 1.httpClient和ScrollView
1 在服务器端使用sqllite编写数据库 常见命令是:sqlite3 tank.db 进入之后创建表: create table tscore ( id integer primary key au ...
- H5、React Native、Native应用对比分析
每日更新关注:http://weibo.com/hanjunqiang 新浪微博!iOS开发者交流QQ群: 446310206 "存在即合理".凡是存在的,都是合乎规律的.任何新 ...
- 上海C++游戏服务器群活动PPT下载
下载页面: http://download.csdn.net/download/jq0123/8227519 跨服与跨区的设计PPT 上海C++游戏服务器群 2014.11.9 沙龙讲义. 自我介绍 ...
- 【一天一道LeetCode】#120. Triangle
一天一道LeetCode 本系列文章已全部上传至我的github,地址:ZeeCoder's Github 欢迎大家关注我的新浪微博,我的新浪微博 欢迎转载,转载请注明出处 (一)题目 Given a ...
- Cocos2D:塔防游戏制作之旅(十二)
以上代码块相当直观 - 但是它分解的有些细致了. 首先,敌人通过传递HelloWorldLayer对象的引用而初始化.在init方法里,少数重要的变量被设置: maxHP:定义敌人有多经打(Tough ...
- (NO.00002)iOS游戏精灵战争雏形(十)
到目前为止,子弹是有去无回.如果子弹击中目标或者飞出屏幕,那么子弹也就没必要存在了. 这里的不存在不是把它从屏幕中删除,因为前面提到了子弹缓存的作用,所以这里仅仅将其设置为不可见就可以了. 首先处理飞 ...
- 使用CocoaPods创建Pod
本来想给App评分,好的开源组件没有Swift版,如是自己写了个简易的.想着既然写了,就写完善点,提供给需要的人使用.这样SwiftyiRate诞生了. 下面主要说下创建pod的步骤: 一.创建git ...
- Leetcode_19_Remove Nth Node From End of List
本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/41778305 Given a linked list, r ...