iOS如何优雅的使用FMDB

//
//  WJDataBaseManager.h
//  wenjin
//
//  Created by Kevin on 2017/4/17.
//  Copyright © 2017年 Hangzhou Chuanwoqi Internet Technology Co., Ltd. All rights reserved.
//

#import <Foundation/Foundation.h>

/**
 数据库管理
 - (void)createTables 中使用model创建表
 */
@interface WJDataBaseManager : NSObject


/**
 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
 在app启动时初始化。例如:[[WJDataBaseManager shareManager] registerTables:@[@"WJTestModel1",@"WJTestModel2"]];
 */
+ (instancetype)shareManager;


/**
 创建表(使用model名创建对应表),model名 -> table名
 注意事项:一个model对应一张表,因此一张表只会被创建一次,model往后的修改不会被重新创建,可使用新model名创建新的table.
 表的设计需慎重,考虑app版本兼容性,建议不要修改model,而使用新的model。

 @param tables 存放model名称的数组。
 */
- (void)registerTables:(NSArray <NSString*>*)tables;

/**
 *  插入数据
 *
 *  param model       模型
 *  param finishBlock 插入完成回调
 */
-(void)insert:(id)model FinishBlock:(void (^)(bool issuccess))finishBlock;
/**
 *  更新数据库
 *
 *  param model 模型
 *  param where 条件
 *  param finishBlock 更新完成回调
 *
 *  return 是否成功
 */
-(void)update:(id)model Where:(NSString*)where FinishBlock:(void (^)(bool issuccess))finishBlock;
/**
 *  查询数据库
 *
 *  param model 模型
 *  param where 条件
 *  param order 排序
 *  param limit 限制行数
 *
 *  return 模型数组
 */
-(NSArray*)select:(id)model Where:(NSString*)where Order:(NSString*)order Limit:(NSString*)limit;

/**
 *  删除
 *
 *  param model 模型
 *  param where 条件
 *
 *  return 删除是否成功
 */
-(BOOL)delete:(id)model Where:(NSString*)where;

/**
 *  查询表总记录数
 *
 *  param model 模型
 *
 *  return 总数
 */
-(NSInteger)getCount:(id)model;
-(NSInteger)getCount:(id)model where:(NSString*)where;
-(NSInteger)getSum:(id)model Filed:(NSString*)filed where:(NSString*)where;
//  表是否存在
-(BOOL)isExit:(id)model;
//  删除表数据
-(void)clearTable:(id)model;

@end
//
//  WJDataBaseManager.m
//  wenjin
//
//  Created by Kevin on 2017/4/17.
//  Copyright © 2017年 Hangzhou Chuanwoqi Internet Technology Co., Ltd. All rights reserved.
//

#import "WJDataBaseManager.h"
#import <MJExtension/MJExtension.h>

#define WJ_DATA_BASE_NAME @"WJ_DATA_BASE_NAME.db"

@interface WJDataBaseManager ()

@property (nonatomic,strong) FMDatabase *db;
@property (nonatomic,strong) FMDatabaseQueue *queue;

@end

@implementation WJDataBaseManager

+(instancetype)shareManager{
    static WJDataBaseManager *instance = nil;
    static dispatch_once_t oneToken;
    dispatch_once(&oneToken, ^{
        instance = [[WJDataBaseManager alloc] init];
    });
    return instance;
}

- (instancetype)init{
    self = [super init];
    if (self) {
        [self createDB];
    }
    return self;
}

#pragma mark -
#pragma mark 创建数据库
- (void)createDB{
    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString *documentDirectory = [paths objectAtIndex:0];
    NSString *dbPath = [documentDirectory stringByAppendingPathComponent:WJ_DATA_BASE_NAME];
    _queue = [FMDatabaseQueue databaseQueueWithPath:dbPath];
    _db = [FMDatabase databaseWithPath:dbPath] ;
    if (![_db open]) {
        NSLog(@"Could not open db.");
        return ;
    }
    NSLog(@"database path : %@",dbPath);
    // 创建表
    [self createTables];
}

- (void)registerTables:(NSArray *)tables{
    for (NSString *table in tables) {
        if(NSClassFromString(table)){
            [self createTableWithModel:NSClassFromString(table)];
        }
    }
}

- (void)createTables{
    
    //根据model创建表
//    [self createTableWithModel:[WJMyStocksModel class]];
    
    
    
}

-(void)removeDatabase{
    [_db close];
    // 直接删除文件
    NSFileManager *fileManager = [NSFileManager defaultManager];
    NSString *dbPath = _db.databasePath;
    if ([fileManager fileExistsAtPath:dbPath]) {
        [fileManager removeItemAtPath:dbPath error:nil];
    }
}

-(void)dropTable:(id)model{
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [NSString stringWithFormat:@"DROP TABLE IF EXISTS %@",table];
    [_db executeUpdate:sql];
}

-(void)clearTable:(id)model{
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [NSString stringWithFormat:@"DELETE FROM %@",table];
    [_db executeUpdate:sql];
}

#pragma mark -
#pragma mark 数据库操作

//  插入数据
-(void)insert:(id)model FinishBlock:(void (^)(bool issuccess))finishBlock{
    NSString *fileds = @"";
    NSArray *keys = [self propertyKeysWithClass:[model class]];
    
    NSString *values = @"";
    for (NSString *key in keys) {
        NSString* value = [NSString stringWithFormat:@"%@",[model valueForKey:key]];
        if ([value isEqualToString:@"(null)"] || !value) {
            value = @"";
        }
        if ([fileds isEqualToString:@""]) {
            fileds = [fileds stringByAppendingFormat:@"%@",key];
            values = [values stringByAppendingFormat:@"'%@'",value];
        }else{
            fileds = [fileds stringByAppendingFormat:@",%@",key];
            values = [values stringByAppendingFormat:@",'%@'",value];
        }
    }
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [[NSString alloc]
                     initWithFormat:@"INSERT INTO %@ (%@) Values(%@)",table,fileds,values];
    // 加入队列
    [_queue inDatabase:^(FMDatabase *dbs){
        bool success = [_db executeUpdate:sql];
        if (finishBlock) {
            finishBlock(success);
        }
    }];
    
}

//  更新数据
-(void)update:(id)model Where:(NSString*)where FinishBlock:(void (^)(bool issuccess))finishBlock{
    
    NSString *fileds = @"";
    NSArray *keys = [self propertyKeysWithClass:[model class]];
    for (NSString *key in keys) {
        NSString* value = [NSString stringWithFormat:@"%@",[model valueForKey:key]];
        if (![value isEqualToString:@"(null)"] && value) {
            if ([fileds isEqualToString:@""]) {
                fileds = [fileds stringByAppendingFormat:@"%@='%@'",key,value];
            }else{
                fileds = [fileds stringByAppendingFormat:@",%@='%@'",key,value];
            }
        }
        
    }
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [[NSString alloc]
                     initWithFormat:@"UPDATE %@ SET %@",table,fileds];
    if (where) {
        sql = [sql stringByAppendingFormat:@" WHERE %@",where];
    }
    // 加入队列
    [_queue inDatabase:^(FMDatabase *dbs){
        bool success = [_db executeUpdate:sql];
        if (finishBlock) {
            finishBlock(success);
        }
    }];
}

//  查询
-(NSArray*)select:(id)model Where:(NSString*)where Order:(NSString*)order Limit:(NSString*)limit{
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [NSString stringWithFormat:@"SELECT * FROM %@",table];
    if (where) {
        sql = [sql stringByAppendingFormat:@" WHERE %@",where];
    }
    if (order) {
        sql = [sql stringByAppendingFormat:@" ORDER BY %@",order];
    }
    if (limit) {
        sql = [sql stringByAppendingFormat:@" LIMIT %@",limit];
    }
    NSMutableArray *array = [NSMutableArray new];
    FMResultSet *rs = [_db executeQuery:sql];
    while ([rs next]) {
        NSDictionary *dic = [rs resultDictionary];
        id m = [[model class] mj_objectWithKeyValues:dic];
        [array addObject:m];
        dic = nil;
        m = nil;
    }
    return array;
}


//  删除
-(BOOL)delete:(id)model Where:(NSString*)where{
    NSString *table = NSStringFromClass([model class]);
    NSString *sql = [NSString stringWithFormat:@"DELETE FROM %@",table];
    if (where) {
        sql = [sql stringByAppendingFormat:@" WHERE %@",where];
    }
    
    BOOL success = [_db executeUpdate:sql];
    return success;
    
}

//  获取表记录总数
-(NSInteger)getCount:(id)model{
    NSString *table = NSStringFromClass([model class]);
    NSInteger count = 0;
    NSString *sqlStr = [NSString stringWithFormat:@"SELECT COUNT(*) FROM %@",table];
    count = [_db intForQuery:sqlStr];
    return count;
}

-(NSInteger)getCount:(id)model where:(NSString*)where{
    NSString *table = NSStringFromClass([model class]);
    NSInteger count = 0;
    NSString *sql = [NSString stringWithFormat:@"SELECT COUNT(*) FROM %@",table];
    if (where) {
        sql = [sql stringByAppendingFormat:@" WHERE %@",where];
    }
    count = [_db intForQuery:sql];
    return count;
}

-(NSInteger)getSum:(id)model Filed:(NSString*)filed where:(NSString*)where{
    NSString *table = NSStringFromClass([model class]);
    NSInteger count = 0;
    NSString *sql = [NSString stringWithFormat:@"SELECT SUM(%@) FROM %@",filed,table];
    if (where) {
        sql = [sql stringByAppendingFormat:@" WHERE %@",where];
    }
    count = [_db intForQuery:sql];
    return count;
}

//  通过模型创建表
-(void)createTableWithModel:(id)model{
    if (![self isExit:model]) {
        NSString *s = [self createSqlWithTableModel:[model class]];
        [_db executeUpdate:s];
    }
}

//  通过模型生成创建表语句
-(NSString*)createSqlWithTableModel:(id)model{
    NSArray *propertys = [self propertyKeysWithClass:[model class]];
    NSString *keys = @"";
    NSString *className = NSStringFromClass([model class]);
    NSString *sql = [NSString stringWithFormat:@"CREATE TABLE %@ (",className];
    for (NSString *key in propertys) {
        //  字符串
        if ([[key class] isSubclassOfClass:[NSString class]]) {
            keys = [keys stringByAppendingString:[NSString stringWithFormat:@"%@ text,",key]];
        }
        //  数字
        if ([[key class] isSubclassOfClass:[NSNumber class]]) {
            keys = [keys stringByAppendingString:[NSString stringWithFormat:@"%@ integer,",key]];
        }
    }
    if ([[keys substringFromIndex:keys.length-1] isEqualToString:@","]) {
        keys = [keys substringToIndex:keys.length-1];
    }
    sql = [sql stringByAppendingString:[NSString stringWithFormat:@"%@)",keys]];
    return sql;
}

//  表是否存在
-(BOOL)isExit:(id)model{
    NSString *table = NSStringFromClass([model class]);
    BOOL exit = NO;
    exit = [_db intForQuery:@"select count(*) as 'count' from sqlite_master where type ='table' and name = ?",table];
    return exit;
}

//  反射对象所有属性
- (NSArray*)propertyKeysWithClass:(Class)classs
{
    unsigned int outCount, i;
    objc_property_t *properties = class_copyPropertyList(classs, &outCount);
    NSMutableArray *keys = [[NSMutableArray alloc] initWithCapacity:outCount];
    for (i = 0; i < outCount; i++) {
        objc_property_t property = properties[i];
        NSString *propertyName = [[NSString alloc] initWithCString:property_getName(property) encoding:NSUTF8StringEncoding];
        [keys addObject:propertyName];
    }
    free(properties);
    return keys;
}

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

推荐阅读更多精彩内容