FMDB简单封装(上拉、下拉、定期清理)

注意: 使用前项目需先pod好FMDB

  1. status.sql (SQL语句类单独一个空文件)
-- 创建数据表 --
CREATE TABLE IF NOT EXISTS "T_Status"(
    "statusId" INTEGER NOT NULL,
    "userId" INTEGER NOT NULL,
    "status" TEXT,
    "createTime" TEXT DEFAULT (datetime('now','localtime')),
    PRIMARY KEY("statusId","userId")
);
  1. CZSQLiteManager.swift (SQLite管理类)

//  Copyright © 2018年 boboMa. All rights reserved.
//

import Foundation
import FMDB

///最大的数据库缓存时间,以 S 为单位 '-5'表示往前推5天
private let maxDBCacheTime:TimeInterval = -5 * 24 * 60 * 60

// MARK: - SQLite管理器
/**
 1.数据库本质上是保存在沙盒中的一个文件,首先需要创建并且打开数据库
   FMDB - 队列
 2.创建数据表
 3.增删改查
 */
class CZSQLiteManager {
    ///单例,全局数据库工具访问点
    static let shared = CZSQLiteManager()
    ///数据库队列
    let queue:FMDatabaseQueue
    
    ///构造函数
    private init(){
       //数据库的全路径 - path
        let dbName = "status.db"
        var  path = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true)[0]
        path = (path as NSString).appendingPathComponent(dbName)
        
        print("数据库的路径" + path)
        //创建数据库队列,同时‘创建或打开’数据库(这时候数据库是空的什么都没有)
        queue = FMDatabaseQueue(path: path)
        // 打开数据库
        createTable()
       //注册通知 - 监听应用程序进入后台
        //模仿 SDWebImage
        NotificationCenter.default.addObserver(self, selector: #selector(clearDBCache), name: NSNotification.Name.UIApplicationDidEnterBackground, object: nil)
    }
    deinit {
        //注销通知
        NotificationCenter.default.removeObserver(self)
    }
    

    
}

// MARK: - 数据操作
extension CZSQLiteManager{
    
// MARK: -*********************************从数据库加载数据数组*********************************
    /// - Parameters:
    ///   - userId: 当前登录的用户账号
    ///   - since_id: 返回ID比since_id大的数据
    ///   - max_id: 返回ID小于max_id的数据
    /// - Returns: 数据的字典的数组,将数据库中status字段对应的二进制数据反序列化,生成字典
    func loadStatus(userId:String,since_id:Int64 = 0,max_id:Int64 = 0) -> [[String:AnyObject]] {
        //1.准备SQL
        var  sql = "SELECT statusId,userId,status FROM T_Status \n"
        
        sql += "WHERE userId = \(userId) \n"
        // 上拉 / 下拉 都是针对同一个ID进行判断
        if since_id > 0{
            //下拉
            sql += "AND statusId > \(since_id) \n"
        }else if max_id > 0{
            //上拉
            sql += "AND statusId < \(max_id) \n"
        }
        sql += "ORDER BY statusId DESC LIMIT 20;"
      //拼接 SQL 结束后,一定要测试
        
        print(sql)
        //2.执行SQL
        let array = execRecordSet(sql: sql)
        //3.遍历数组,将数组中的 status 反序列化 -> 字典的数组
        var result = [[String:AnyObject]]()
        for dict in array {
            //反序列化
            guard let jsonData = dict["status"] as? Data,
                let json = try? JSONSerialization.jsonObject(with: jsonData, options: []) as?[String:AnyObject]else{
                    continue
            }
            //追加到数组
            result.append(json ?? [:])
        }
        
        
        return result
    }
    
    
    
// MARK: -*********************************新增或者修改**************************************
    /**
     从网络加载结束后,返回的是数据的‘字典数组’,每一个字典对应一个完整的数据记录
      - 完整的数据记录中,包含数据的代号
      - 数据记录中,没有当前登录的用户代号
     */
  
    /// 新增或者修改数据数据,数据数据在刷新的时候,可能会出现重叠
    ///   - userId: 当前登录用户的ID
    ///   - array: 从网络获取的‘字典数组’
    func updateStatus(userId:String,array:[[String:AnyObject]]){
       
        //1.准备SQL
        /**
         statusId: 要保存的数据代号
         userId:   当前登录用户的 ID
         status:  完整数据字典的json 二进制数据
         */
        let sql = "INSERT OR REPLACE INTO T_Status(statusId,userId,status) VALUES(?,?,?);"
        
        //2.执行SQL
        
        queue.inTransaction { (db, rollback) in
            
            //遍历数组,逐条插入数据数据
            for dict in array{
                //从字典获取数据代号 / 将字典序列化成二进制数据
                guard let statusId = dict["idstr"] as? String,
                      let jsonData = try?  JSONSerialization.data(withJSONObject: dict, options: []) else{
                        continue
                }
                
                //执行 SQL
                if db.executeUpdate(sql, withArgumentsIn: [statusId,userId,jsonData]) == false{
                    //需要回滚 *rollback = YES;
                    //Swift 1.x & 2.x => rollback.memory = true
                    //Swift3.0的写法
                    rollback.pointee = true
                    
                    break
                }
            }
        }
    }
// MARK: -*********************************清理数据缓存*********************************
    ///清理数据缓存
    ///注意细节:
    ///SQLite 随着数据不断的增加,数据库文件的大小会不断的增加
    ///但是: 如果删除了数据,数据库大小,不会变小!
    ///如果要变小
    ///1>将数据库文件复制一个新的副本,status.db.old
    ///2>新建一个空的数据库文件
    ///3>自己编写SQL,从old中将所有数据读出写入新的数据库
    @objc private func clearDBCache(){
        let dateString = Date.cz_dateString(delta: maxDBCacheTime)
        print("清理数据缓存\(dateString)")
        //准备 SQL
        let sql = "DELETE FROM T_Status WHERE createTime < ?;"
        //执行 SQL
        queue.inDatabase { (db) in
            if db.executeUpdate(sql, withArgumentsIn: [dateString]) == true{
                print("删除了\(db.changes )条记录")
            }
        }
        
        
        
        
    }
}

//MARK: - *********************************创建数据表以及其他私有方法*********************************
 extension CZSQLiteManager{
   
//MARK: -*********************************查询方法*********************************
    func execRecordSet(sql:String) -> [[String:AnyObject]] {
        //结果数组
        var result = [[String:AnyObject]]()
        queue.inDatabase { (db) in
            guard let rs = db.executeQuery(sql, withArgumentsIn: []) else{
                return
            }
            //逐行遍历结果集合
            while rs.next(){
                //1>列数
                let colCount = rs.columnCount
                
                
                //2.>遍历所有列
                for col in 0..<colCount{
                    //3>列名 - KEY
                    //   值 -> Value
                  guard  let name = rs.columnName(for: col),
                    let value = rs.object(forColumnIndex: col)else{
                        continue
                    }
                    
                    //4>追加结果
                    result.append([name : value as AnyObject])
                   // print(name,value)
                }
            }
        }
        return result
    }
    
    
  //MARK: -*********************************创建数据表*********************************
    func createTable() {
        //1.SQL
        guard let path = Bundle.main.path(forResource: "status.sql", ofType: nil),
              let sql = try? String.init(contentsOfFile: path)//创建表的SQL语句
            else {
            return
        }
        print(sql)
        //2.执行 SQL - FMDB 的内部队列,串行队列,同步执行
        queue.inDatabase { (db) in
            if db.executeStatements(sql) == true{
                print("创表成功")
            }else{
                print("创表失败")
            }
        }
    }
}


  1. Date+Extensions.swift (日期分类)
//  Copyright © 2018年 boboMa. All rights reserved.
//

import Foundation
///日期格式化器 - 不要频繁的释放和创建,会影响性能
private let dateFormatter = DateFormatter()

extension Date{
    ///计算与当前系统时间偏差 delta 秒数的日期字符串
    ///在 Swift中,如果要定义结构体的’类‘函数,使用 static 修饰 -> 静态函数
    
    static func cz_dateString(delta:TimeInterval)->String{
        let date = Date(timeIntervalSinceNow: delta)
    //指定日期格式
        dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss"
    return dateFormatter.string(from: date)
    }
}


  1. 项目下载地址--https://github.com/Bobo168/DataBase
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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