這篇文章主要介紹IOS數(shù)據(jù)庫(kù)升級(jí)數(shù)據(jù)遷移的示例分析,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!
創(chuàng)新互聯(lián)公司-專(zhuān)業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比三門(mén)網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式三門(mén)網(wǎng)站制作公司更省心,省錢(qián),快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋三門(mén)地區(qū)。費(fèi)用合理售后完善,十載實(shí)體公司更值得信賴。
IOS 數(shù)據(jù)庫(kù)升級(jí)數(shù)據(jù)遷移的實(shí)例詳解
概要:
很久以前就遇到過(guò)數(shù)據(jù)庫(kù)版本升級(jí)的引用場(chǎng)景,當(dāng)時(shí)的做法是簡(jiǎn)單的刪除舊的數(shù)據(jù)庫(kù)文件,重建數(shù)據(jù)庫(kù)和表結(jié)構(gòu),這種暴力升級(jí)的方式會(huì)導(dǎo)致舊的數(shù)據(jù)的丟失,現(xiàn)在看來(lái)這并不不是一個(gè)優(yōu)雅的解決方案,現(xiàn)在一個(gè)新的項(xiàng)目中又使用到了數(shù)據(jù)庫(kù),我不得不重新考慮這個(gè)問(wèn)題,我希望用一種比較優(yōu)雅的方式去解決這個(gè)問(wèn)題,以后我們還會(huì)遇到類(lèi)似的場(chǎng)景,我們都想做的更好不是嗎?
理想的情況是:數(shù)據(jù)庫(kù)升級(jí),表結(jié)構(gòu)、主鍵和約束有變化,新的表結(jié)構(gòu)建立之后會(huì)自動(dòng)的從舊的表檢索數(shù)據(jù),相同的字段進(jìn)行映射遷移數(shù)據(jù),而絕大多數(shù)的業(yè)務(wù)場(chǎng)景下的數(shù)據(jù)庫(kù)版本升級(jí)是只涉及到字段的增減、修改主鍵約束,所以下面要實(shí)現(xiàn)的方案也是從最基本的、最常用的業(yè)務(wù)場(chǎng)景去做一個(gè)實(shí)現(xiàn),至于更加復(fù)雜的場(chǎng)景,可以在此基礎(chǔ)上進(jìn)行擴(kuò)展,達(dá)到符合自己的預(yù)期的。
選型定型
網(wǎng)上搜索了下,并沒(méi)有數(shù)據(jù)庫(kù)升級(jí)數(shù)據(jù)遷移簡(jiǎn)單完整的解決方案,找到了一些思路
1.清除舊的數(shù)據(jù),重建表
優(yōu)點(diǎn):簡(jiǎn)單
缺點(diǎn):數(shù)據(jù)丟失
2.在已有表的基礎(chǔ)上對(duì)表結(jié)構(gòu)進(jìn)行修改
優(yōu)點(diǎn):能夠保留數(shù)據(jù)
缺點(diǎn):規(guī)則比較繁瑣,要建立一個(gè)數(shù)據(jù)庫(kù)的字段配置文件,然后讀取配置文件,執(zhí)行SQL修改表結(jié)構(gòu)、約束和主鍵等等,涉及到跨多個(gè)版本的數(shù)據(jù)庫(kù)升級(jí)就變得繁瑣并且麻煩了
3.創(chuàng)建臨時(shí)表,把舊的數(shù)據(jù)拷貝到臨時(shí)表,然后刪除舊的數(shù)據(jù)表并且把臨時(shí)表設(shè)置為數(shù)據(jù)表。
優(yōu)點(diǎn):能夠保留數(shù)據(jù),支持表結(jié)構(gòu)的修改,約束、主鍵的變更,實(shí)現(xiàn)起來(lái)比較簡(jiǎn)單
缺點(diǎn):實(shí)現(xiàn)的步驟比較多
綜合考慮,第三種方法是一個(gè)比較靠譜的方案。
主要步驟
根據(jù)這個(gè)思路,分析了一下數(shù)據(jù)庫(kù)升級(jí)了主要步驟大概如下:
獲取數(shù)據(jù)庫(kù)中舊的表
修改表名,添加后綴“_bak”,把舊的表當(dāng)做備份表
創(chuàng)建新的表
獲取新創(chuàng)建的表
遍歷舊的表和新表,對(duì)比取出需要遷移的表的字段
數(shù)據(jù)遷移處理
刪除備份表
使用到的SQL語(yǔ)句分析
這些操作都是和數(shù)據(jù)庫(kù)操作有關(guān)系的,所以問(wèn)題的關(guān)鍵是對(duì)應(yīng)步驟的SQL語(yǔ)句了,下面分析下用到的主要的SQL語(yǔ)句:
獲取數(shù)據(jù)庫(kù)中舊的表
SELECT * from sqlite_master WHERE type='table'
結(jié)果如下,可以看到有type | name | tbl_name | rootpage | sql 這些數(shù)據(jù)庫(kù)字段,我們只要用到name也就是數(shù)據(jù)庫(kù)名稱這個(gè)字段就行了
sqlite> SELECT * from sqlite_master WHERE type='table' ...> ; +-------+---------------+---------------+----------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ | type | name | tbl_name | rootpage | sql | +-------+---------------+---------------+----------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ | table | t_message_bak | t_message_bak | 2 | CREATE TABLE "t_message_bak" (messageID TEXT, messageType INTEGER, messageJsonContent TEXT, retriveTimeString INTEGER, postTimeString INTEGER, readState INTEGER, PRIMARY KEY(messageID)) | | table | t_message | t_message | 4 | CREATE TABLE t_message ( messageID TEXT, messageType INTEGER, messageJsonContent TEXT, retriveTimeString INTEGER, postTimeString INTEGER, readState INTEGER, addColumn INTEGER, PRIMARY KEY(messageID) ) | +-------+---------------+---------------+----------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 2 行于數(shù)據(jù)集 (0.03 秒)
修改表名,添加后綴“_bak”,把舊的表當(dāng)做備份表
-- 把t_message表修改為t_message_bak表 ALTER TABLE t_message RENAME TO t_message_bak
獲取表字段信息
-- 獲取t_message_bak表的字段信息 PRAGMA table_info('t_message_bak')
獲取到的表字段信息如下,可以看到有| cid | name | type | notnull | dflt_value | pk | 這些數(shù)據(jù)庫(kù)字段,我們只要用到name也就是字段名稱這個(gè)字段就行了
sqlite> PRAGMA table_info('t_message_bak'); +------+--------------------+---------+---------+------------+------+ | cid | name | type | notnull | dflt_value | pk | +------+--------------------+---------+---------+------------+------+ | 0 | messageID | TEXT | 0 | NULL | 1 | | 1 | messageType | INTEGER | 0 | NULL | 0 | | 2 | messageJsonContent | TEXT | 0 | NULL | 0 | | 3 | retriveTimeString | INTEGER | 0 | NULL | 0 | | 4 | postTimeString | INTEGER | 0 | NULL | 0 | | 5 | readState | INTEGER | 0 | NULL | 0 | +------+--------------------+---------+---------+------------+------+ 6 行于數(shù)據(jù)集 (0.01 秒)
使用子查詢進(jìn)行數(shù)據(jù)遷移處理
INSERT INTO t_message(messageID, messageType, messageJsonContent, retriveTimeString, postTimeString, readState) SELECT messageID, messageType, messageJsonContent, retriveTimeString, postTimeString, readState FROM t_message_bak
把t_message_bak表中的messageID, messageType, messageJsonContent, retriveTimeString, postTimeString, readState這些字段的值復(fù)制到t_message表中
代碼實(shí)現(xiàn)
接下來(lái)就到了代碼的實(shí)現(xiàn)步驟了
// 創(chuàng)建新的臨時(shí)表,把數(shù)據(jù)導(dǎo)入臨時(shí)表,然后用臨時(shí)表替換原表 - (void)baseDBVersionControl { NSString * version_old = ValueOrEmpty(MMUserDefault.dbVersion); NSString * version_new = [NSString stringWithFormat:@"%@", DB_Version]; NSLog(@"dbVersionControl before: %@ after: %@",version_old,version_new); // 數(shù)據(jù)庫(kù)版本升級(jí) if (version_old != nil && ![version_new isEqualToString:version_old]) { // 獲取數(shù)據(jù)庫(kù)中舊的表 NSArray* existsTables = [self sqliteExistsTables]; NSMutableArray* tmpExistsTables = [NSMutableArray array]; // 修改表名,添加后綴“_bak”,把舊的表當(dāng)做備份表 for (NSString* tablename in existsTables) { [tmpExistsTables addObject:[NSString stringWithFormat:@"%@_bak", tablename]]; [self.databaseQueue inDatabase:^(FMDatabase *db) { NSString* sql = [NSString stringWithFormat:@"ALTER TABLE %@ RENAME TO %@_bak", tablename, tablename]; [db executeUpdate:sql]; }]; } existsTables = tmpExistsTables; // 創(chuàng)建新的表 [self initTables]; // 獲取新創(chuàng)建的表 NSArray* newAddedTables = [self sqliteNewAddedTables]; // 遍歷舊的表和新表,對(duì)比取出需要遷移的表的字段 NSDictionary* migrationInfos = [self generateMigrationInfosWithOldTables:existsTables newTables:newAddedTables]; // 數(shù)據(jù)遷移處理 [migrationInfos enumerateKeysAndObjectsUsingBlock:^(NSString* newTableName, NSArray* publicColumns, BOOL * _Nonnull stop) { NSMutableString* colunmsString = [NSMutableString new]; for (int i = 0; i* migrationInfos = [NSMutableDictionary dictionary]; for (NSString* newTableName in newTables) { NSString* oldTableName = [NSString stringWithFormat:@"%@_bak", newTableName]; if ([oldTables containsObject:oldTableName]) { // 獲取表數(shù)據(jù)庫(kù)字段信息 NSArray* oldTableColumns = [self sqliteTableColumnsWithTableName:oldTableName]; NSArray* newTableColumns = [self sqliteTableColumnsWithTableName:newTableName]; NSArray* publicColumns = [self publicColumnsWithOldTableColumns:oldTableColumns newTableColumns:newTableColumns]; if (publicColumns.count > 0) { [migrationInfos setObject:publicColumns forKey:newTableName]; } } } return migrationInfos; } - (NSArray*)publicColumnsWithOldTableColumns:(NSArray*)oldTableColumns newTableColumns:(NSArray*)newTableColumns { NSMutableArray* publicColumns = [NSMutableArray array]; for (NSString* oldTableColumn in oldTableColumns) { if ([newTableColumns containsObject:oldTableColumn]) { [publicColumns addObject:oldTableColumn]; } } return publicColumns; } - (NSArray*)sqliteTableColumnsWithTableName:(NSString*)tableName { __block NSMutableArray * tableColumes = [NSMutableArray array]; [self.databaseQueue inDatabase:^(FMDatabase *db) { NSString* sql = [NSString stringWithFormat:@"PRAGMA table_info('%@')", tableName]; FMResultSet *rs = [db executeQuery:sql]; while ([rs next]) { NSString* columnName = [rs stringForColumn:@"name"]; [tableColumes addObject:columnName]; } }]; return tableColumes; } - (NSArray*)sqliteExistsTables { __block NSMutableArray * existsTables = [NSMutableArray array]; [self.databaseQueue inDatabase:^(FMDatabase *db) { NSString* sql = @"SELECT * from sqlite_master WHERE type='table'"; FMResultSet *rs = [db executeQuery:sql]; while ([rs next]) { NSString* tablename = [rs stringForColumn:@"name"]; [existsTables addObject:tablename]; } }]; return existsTables; } - (NSArray*)sqliteNewAddedTables { __block NSMutableArray * newAddedTables = [NSMutableArray array]; [self.databaseQueue inDatabase:^(FMDatabase *db) { NSString* sql = @"SELECT * from sqlite_master WHERE type='table' AND name NOT LIKE '%_bak'"; FMResultSet *rs = [db executeQuery:sql]; while ([rs next]) { NSString* tablename = [rs stringForColumn:@"name"]; [newAddedTables addObject:tablename]; } }]; return newAddedTables; }
以上是“IOS數(shù)據(jù)庫(kù)升級(jí)數(shù)據(jù)遷移的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!