当前位置: 首页 > web >正文

【iOS(swift)笔记-11】App版本升级时本地数据库sqlite更新逻辑

 

/*
     要操作文件需要将"项目名.entitlements"文件里配置的 App Sandbox 设置为 No
     */
    
    /*
     
     原理描述:
     Resources资源同名文件在包更新时会被替换掉,所以为了保留用户本地已有数据,数据库需要采用不同文件名进行操作,具体如下
     
     static let dbName = "db.db" // 原始数据库,应用的原始空数据库
     static let dbLocalName = "dbLocal.db" // 用户本地数据库,就是db.db的复制品,只是记录了用户的数据
     static let dbLocalTempName = "dbLocalTemp.db" // 更新数据库时用的临时数据库
     // 数据库版本(数据库表中设置一个字段dbVersion,可用数字(比如1或字符串1.0.0)存储,记录当前数据库的版本。用于判断数据库是否已更新)
     
     在版本升级的时候,为了保证用户已有数据,而且新数据库表结构字段可能增删改,不能随随便便替换数据库,而是要合理地“转移”数据。
     
   (用数据库版本为标识是否已更新数据库)
   数据库更新处理方式:
   第1步,判断是否已更新过数据库(拿db.db和dbLocal.db中的dbVersion做比较,如果已更新过了就不用执行后续步骤)
   第2步,原始数据库db.db->(直接复制一份)->临时数据库dbLocalTemp.db
   第3步,本地数据库dbLocal.db->(用SQL查询语句将用户数据复制到)->临时数据库dbLocalTemp.db
     【因为新的数据库可能增加了属性或新的表,所以要以新的数据库为中心,将数据填充完整后替换。而不是在旧的正式数据库做修改,这样很麻烦而且容易遗漏】
   第4步,删除本地数据库dbLocal.db
   第5步,临时数据库dbLocalTemp.db改名为dbLocal.db

   数据库表的设计越复杂,具体复制数据的脚步就会越麻烦,所以最好是用服务器数据库存储数据,这样就算数据库改动也方便操作。本地数据库没事就别瞎改动了。

   */

/*注,操作SQLite数据库采用第三方SQLite.swifthttps://gitter.im/stephencelis/SQLite.swift*/import AppKit
import SQLiteclass DBControl {static func initializeDB() {/* 这里本人将数据库文件放到了项目的资源根目录下(与Info.plist同级) */let dbPath = filePath(inResourceDirectory: Config.dbName)let dbLocalPath = filePath(inResourceDirectory: Config.dbLocalName)let dbLocalTempPath = filePath(inResourceDirectory: Config.dbLocalTempName)if (!fileExists(inResourceDirectory: Config.dbLocalName)) {print("没有本地数据库")// 没有数据库,直接复制数据库过去do {try FileManager.default.copyItem(atPath: dbPath, toPath: dbLocalPath)print("复制成功")} catch {print("复制失败")}}else {// 有数据库print("有数据库");// 判断是否已更新var dbVersion:String = ""var dbLocalVersion:String = ""do {let dbCon = try Connection(dbPath)let dbLocalCon = try Connection(dbLocalPath)dbVersion = try dbCon.scalar("SELECT value FROM Records where 1=1 and name='DBVersion'") as! String // 根据自己创建的表结构进行查询dbLocalVersion = try dbLocalCon.scalar("SELECT value FROM Records where 1=1 and name='DBVersion'") as! String} catch {}print("dbLocalVersion=\(dbLocalVersion)")print("dbVersion=\(dbVersion)")if (Int(dbLocalVersion)  ?? 0 >= Int(dbVersion) ?? 0) {// 已更新过了,无需再更新print("已更新过了,无需再更新");}else {// 需要更新print("需要更新");// 复制临时数据库do {if (fileExists(inResourceDirectory: Config.dbLocalTempName)) {try FileManager.default.removeItem(at: URL(fileURLWithPath: dbLocalTempPath)) // 如果有先删掉}try FileManager.default.copyItem(atPath: dbPath, toPath: dbLocalTempPath)print("临时数据库复制成功")} catch {}// 更新数据(!!!根据表的变动情况,灵活采用更新的方式)do {// 1、复制数据// ↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓这里有一大堆代码要写↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓// 将dbLocal.db内表的数据复制到dbLocalTemp.db// 就是写一大堆SQL语句执行// ↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑这里有一大堆代码要写↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑// 2、删除本地数据库dbLocal.dbtry FileManager.default.removeItem(at: URL(fileURLWithPath: dbLocalPath))// 3、临时数据库dbLocalTemp.db改名为dbLocal.dbtry FileManager.default.moveItem(at: URL(fileURLWithPath: dbLocalTempPath), to: URL(fileURLWithPath: dbLocalPath))} catch {}}}}static func filePath(inResourceDirectory resourceName: String) -> String {/*注意!!!1、直接用Bundle.main.path(forResource: "", ofType: "")来获取路径的话,如果文件不存在则返回的结果为nil,不是想要的路径字符串2、刚刚新动态创建的文件,明明确确实实已经存在于那里了,立马用Bundle.main.path(forResource: "", ofType: "")去获取路径,得到的结果却是nil,甚是离谱*///// 获取应用程序的bundleif let bundlePath = Bundle.main.resourceURL?.path {// 构建资源目录的完整路径
//            let resourceDirectoryPath = URL(fileURLWithPath: bundlePath).appendingPathComponent("Resources").pathlet resourceDirectoryPath = URL(fileURLWithPath: bundlePath).appendingPathComponent("").path // (以根目录为基)其下具体哪个文件夹路径,这里不写,写在参数resourceName里// 构建要检查的文件路径let filePath = URL(fileURLWithPath: resourceDirectoryPath).appendingPathComponent(resourceName).pathreturn filePath}return ""}static func fileExists(inResourceDirectory resourceName: String) -> Bool {// 获取应用程序的bundleif let bundlePath = Bundle.main.resourceURL?.path {// 构建资源目录的完整路径
//            let resourceDirectoryPath = URL(fileURLWithPath: bundlePath).appendingPathComponent("Resources").pathlet resourceDirectoryPath = URL(fileURLWithPath: bundlePath).appendingPathComponent("").path //  (以根目录为基)其下具体哪个文件夹路径,这里不写,写在参数resourceName里// 构建要检查的文件路径let filePath = URL(fileURLWithPath: resourceDirectoryPath).appendingPathComponent(resourceName).path// 使用FileManager检查文件是否存在let fileManager = FileManager.defaultvar isDirectory: ObjCBool = falselet fileExists = fileManager.fileExists(atPath: filePath, isDirectory: &isDirectory)
//            print("filePath=\(filePath)")// 返回文件是否存在return fileExists && !isDirectory.boolValue // 确保不是目录}return false}
}

http://www.xdnf.cn/news/9221.html

相关文章:

  • 二十九、面向对象底层逻辑-SpringMVC九大组件之MultipartResolver接口设计
  • leetcode每日一题 -- 2131.连接两字母单词得到的最长回文串
  • taro + vue3 实现小程序sse长连接实时对话
  • el-tree拖拽事件,限制同级拖拽,获取拖拽后节点的前后节点,同级拖拽合并父节点name且子节点加入目标节点里
  • 让 Deepseek 写一个尺码计算器
  • LLM 驱动的 Go 到 Rust 项目迁移的挑战与实践
  • PHP生成pdf方法
  • AJAX 数据库
  • 第四十五篇-Tesla P40+Qwen3-30B-A3B部署与测试
  • Linux Shell 切换
  • Transformer 通关秘籍10:词向量运算:queen=king-man+wowem
  • 2025年5月6日 飞猪Java一面
  • 2025 年江西研究生数学建模竞赛题C题基于大雾背景视频学习的能见度回归建模完整思路 模型代码 结果 成品分享
  • 为(FramePack)的视频生成添加首尾帧功能
  • OpenGL Chan视频学习-11 Uniforms in OpenGL
  • 【洛谷P9303题解】AC- [CCC 2023 J5] CCC Word Hunt
  • 功耗仅4W!迷你服务器黑豹X2(Panther X2)卡刷、线刷刷入Armbian(ubuntu)系统教程
  • 鸿蒙OSUniApp 制作美观的文章列表展示组件#三方框架 #Uniapp
  • 11.12 LangGraph全局共享状态实战:200ms实现50+仓库AI协同,效率飙升!
  • vscode的Embedded IDE创建keil项目找不到源函数或者无法跳转
  • windows中Redis、MySQL 和 Elasticsearch启动并正确监听指定端口
  • 亚马逊服务器磁盘扩容一般操作
  • 基于springboot的校园商铺管理系统的设计与实现
  • 大型三甲医院更换HIS系统全流程分析与经验考察(下)
  • 【React】-组件中实现高性能鼠标跟随提示框的完整优化过程
  • AI赋能引爆短剧全球化风潮,腾讯云媒体处理助力短剧平台出海吸金
  • 中国免税品人工智能商城:引领免税品市场新潮流
  • transformer总结
  • 华为OD机试真题——斗地主之顺子(2025B卷:100分)Java/python/JavaScript/C/C++/GO最佳实现
  • QAtomicInt原子变量的CAS(Compare And Swap)写法与优缺点