iOS-FMDB本地存储之一种封装思想

前言

家里养的是哈士奇,它小的时候,我总喜欢抱着它睡.后来长大了,发现床越来越挤了,就不让他上床了,这货叫的比杀猪还惨,没办法我只好陪它打地铺抱着它睡,让它习惯.可是当我醒来的时候,次噢!这货为嘛在我床上.
不开心的时候就看的笑话,还记得小时候家里的德国黑贝叫贝贝,长得特别大,平时把它拴上怕它咬人,有时把它放开,它没有立即跑开去玩,而是拼命的去撕咬栓它的链子,当时看了觉得很好笑.它已经离开我们好久了,知道自己得了病,自己找个没人的地方安静的死去.


好想养一条呀,应该快了

1.前提工作

安装cocoapods,并且导入

pod 'FMDB/FTS', '2.5'               # sqlite操作
pod 'FCFileManager', '1.0.17'       # sqlite操作
pod 'GCJSONKit', '1.5.0'            # JSON和对象互转

2.创建可视化表

首先创建.bundle文件用来装所有表(便于管理)


bundle.png

然后在.string里面规定自己表定义格式如下图:
表名:user_friends
主键:USER_CODE
字段:数组


table.png

3.加载表定义

创建一个manager 来管理表,加载表定义,需要传入你.bundle文件的名字和数据库存储路径.

/**
 * 初始化数据对象
 * @param defineFileName :  数据库定义文件
 * @param  filePath : 数据库文件路径
 */
- (instancetype)initWithDBDefine:(NSString * )bundleName filePath:(NSString *)filePath {
    if (self =[super init]) {
        self.dbFile = filePath;
        [self loadTableDefinition:bundleName];
    }
    return self;
}

4.创建表

/**
 * 根据定义创建数据库表
 *
 */
- (BOOL)createTableWithConfig:(NSDictionary *)tableDefine withDb:(FMDatabase *)db

数据库创建表具体sql语句

CREATE TABLE IF NOT EXISTS user_friends(USER_CODE TEXT primary key not null,USER_NAME TEXT,FRIEND_CODE TEXT,USER_IMG_SRC TEXT,USER_SHORT_NAME TEXT,USER_EN_NAME TEXT,USER_SEX TEXT,DEPT_NAME TEXT,USER_POST TEXT)

你所需要的是表名,主键,及各种字段名字拼成以上格式,这里不多说可以文章底部github下载demo自己研究这里不多说

5.打开数据库的管理类

在这创建manger,并传入存储路径
保证数据库,操作在同一线程操作,当我们在程序中运用到多线程的时候,那么你必须要考虑的就是各线程抢占资源的问题,不能让同一时间多个线程去抢一个资源,比如你两个线程同时去操作sql,就会造成有脏读数据或者查不到数据,或者查的是脏数据.

#import "SQliteUser.h"

@interface SQliteUser ()

@property (nonatomic, strong) SqliteManager *manager;

@end

@implementation SQliteUser

SINGLETON_FOR_CLASS(SQliteUser);//宏单例

- (SqliteManager *)manager {
    if (_manager) {
        return _manager;
    }
    @synchronized(self) {
        if (!_manager) {
             NSString *dbPath = [NSString stringWithFormat:@"%@%@",NSHomeDirectory(),@"/Documents/yuhechuan.db"];
            _manager = [[SqliteManager alloc] initWithDBDefine:@"user" filePath:dbPath];
        }
    }
    return _manager;
}

@end

synchronized同步锁,保证一个线程执行完成,再执行其他线程任务
一般说synchronized是加锁,或者说是加对象锁,其实对象锁只是synchronized在实现锁机制中的一种锁(重量锁,用这种方式互斥线程开销大所以叫重量锁,或者叫对象monitor),而synchronized的锁机制会根据线程竞争情况在运行会有偏向锁、轻量锁、对象锁,自旋锁(或自适应自旋锁)等
参考链接:http://www.jianshu.com/p/5dbb07c8d5d5

6.创建DAO

@implementation SQBaseDAO

子类继承baseDAO 实现下面两个父类的方法,来告诉manger,表名和主键

-(NSString *)getPK {
    return @"需要子类实现";
}
- (NSString *)getTable {
    return @"需要子类实现";
}

获取Sqlite管理类和其中的FMDatabaseQueue

- (SqliteManager *)getDataSource {
    return [[SQliteUser sharedInstance] manager];
}

7.实现增,删,改,查

首先调用数据库的方法应在FMDB提供的方法里面执行

- (void)inDatabase:(void (^)(FMDatabase *db))block

这个方法提供了一个代码块。操作数据库的代码写在block里,如:
FMDatabaseQueue是一个串行队列,它不支持串行任务嵌套执行
以下代码比较简单不予多说自己看

//增
- (BOOL)insert:(NSDictionary *)data {
    //安全线程
    FMDatabaseQueue *dbQueue = [[self getDataSource] dbQueue];
    __block BOOL execute = NO;
    [dbQueue inDatabase:^(FMDatabase *db) {
        NSMutableDictionary *mData = [NSMutableDictionary dictionaryWithDictionary:data];
        NSString *pk = [self getPK];//主键
        NSString * table = [self getTable];//表名
        //如果主键没值,主动生成主键
        if (![mData objectForKey:pk]) {
            [mData setObject:[[NSUUID UUID] UUIDString] forKey:pk];
        }
        NSMutableString *insertKey = [NSMutableString stringWithCapacity:0];
        NSMutableString *insertValuesString = [[NSMutableString alloc] init];
        NSMutableArray *insertValues = [[NSMutableArray alloc] init];
        NSArray *columnArray = [mData allKeys];
        for (int i = 0; i < columnArray.count; i++) {
            NSString *columnName = columnArray[i];
            id value = [mData objectForKey:columnName];
            if (!value) {
                continue;
            }
            if (insertKey.length > 0) {
                [insertKey appendString:@","];
                [insertValuesString appendString:@","];
            }
            
            [insertKey appendString:columnName];
            [insertValuesString appendString:@"?"];
            
            [insertValues addObject:value];
        }
        // 拼接insertSQL 语句  采用 replace 插入
        NSString *insertSQL = [NSString stringWithFormat:@"replace into %@(%@) values(%@)", table, insertKey, insertValuesString];
        execute = [db executeUpdate:insertSQL withArgumentsInArray:insertValues];
        //打印日志,如果在主线程执行,方便查找
        [self printSQLLog:insertSQL values:insertValues];
        
    }];
    return execute;
}

//删
- (BOOL)remove:(SQConditionBean *)condition {
    FMDatabaseQueue *dbQueue = [[self getDataSource] dbQueue];
    __block BOOL execute = NO;
    [dbQueue inDatabase:^(FMDatabase *db) {
        NSString *table = [self getTable];
        NSMutableString *deleteSQL = [NSMutableString stringWithFormat:@"delete from %@  ", table];
        // 添加where 语句
        NSMutableArray *valuearray = [NSMutableArray array];
        //获取条件字符串
        NSDictionary *dict = [condition conditionDict];
        NSString *sqlwhere = [self dictionaryToSqlWhere:dict andValues:valuearray];
        if (sqlwhere.length > 0) {
            [deleteSQL appendString:@" where "];
            [deleteSQL appendString:sqlwhere];
        }
        execute = [db executeUpdate:deleteSQL withArgumentsInArray:valuearray];
        //打印日志,如果在主线程执行,方便查找
        [self printSQLLog:deleteSQL values:valuearray];

    }];
    return execute;
}

//改
- (BOOL)modify:(NSDictionary *)data {
    FMDatabaseQueue *dbQueue = [[self getDataSource ] dbQueue];
    __block BOOL execute = NO;
    [dbQueue inDatabase:^(FMDatabase *db) {
        NSString *pk = [self getPK];//主键
        NSString * table = [self getTable];//表名
        NSMutableString *updateKey = [NSMutableString string];//更新key集合
        NSMutableArray *updateValues = [[NSMutableArray alloc] init];//更新的value集合
        NSArray *columnArray = data.allKeys;
        for (NSString *key in columnArray) {
            //不是主键字段
            if (![key isEqualToString:pk]) {
                if (updateKey.length > 0) {
                    [updateKey appendString:@","];
                }
                [updateKey appendFormat:@"%@=?", key];
            }
            id value = [data objectForKey:key];
            [updateValues addObject:value];
        }
        NSMutableString *updateSQL = [NSMutableString stringWithFormat:@"update %@ set %@ where %@=?", table, updateKey,[updateValues lastObject]];
        execute = [db executeUpdate:updateSQL withArgumentsInArray:updateValues];
        //打印日志,如果在主线程执行,方便查找
        [self printSQLLog:updateSQL values:updateValues];
    }];
    return execute;
}

//查
- (NSMutableArray *)query:(SQConditionBean *)condition{
    FMDatabaseQueue *dbQueue = [[self getDataSource ] dbQueue];
    __block NSMutableArray *results = [[NSMutableArray alloc] init];
    [dbQueue inDatabase:^(FMDatabase *db) {
        NSString * table = [self getTable];
        NSString *columnsString = @"*";
        
        //build query
        NSMutableString *query = [NSMutableString stringWithFormat:@"select %@ from %@", columnsString, table];
        NSMutableArray *whereValues = [NSMutableArray array];
        
        //build where
        NSString *wherekey = [self dictionaryToSqlWhere:[condition conditionDict] andValues:whereValues];
        if ( wherekey.length > 0) {
            [query appendFormat:@" where %@", wherekey];
        }
        
        //execute
        FMResultSet *set = nil;
        if (whereValues.count == 0) {
            set = [db executeQuery:query];
        } else {
            set = [db executeQuery:query withArgumentsInArray:whereValues];
        }
        //打印日志,如果在主线程执行,方便查找
        [self printSQLLog:query values:whereValues];
        
        
        while ([set next]) {
            NSDictionary *rowData = [self resultToDic:set];
            [results addObject:rowData];
        }
        [set close];

    }];
    return results;
}

8.创建条件SQConditionBean 拼接where语句

携带一个字典

@property (nonatomic, strong) NSMutableDictionary *conditionDict;
/**
 *  添加条件(操作符是AND)
 *
 *  @param value 值
 *  @param key   字段
 */
-(void) set:(id)value forKey:(NSString *)key {
    if (value) {
        [conditionDict setValue:value forKey:key];
    }
}
/**
 *  添加  != 条件 (操作符是AND)
 *
 *  @param value 值
 *  @param key   字段
 */
-(void) andNE:(id)value forKey:(NSString *)key {
    NSString * query =  [NSString stringWithFormat:@"%@ != ", key];
    [conditionDict setValue:value forKey:query];
}

参考资料:http://www.jianshu.com/p/5dbb07c8d5d5
Demo下载地址:https://github.com/yuhechuan/SqliteTools

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

推荐阅读更多精彩内容