一般app啟動之後,都有一個初始化的過程。此外後續app升級,還需要考慮數據遷移。所以初始化和數據遷移的框架,在初期的版本就要考慮好
總結一下我們的app采取的方案:
1、在持久化的文件夾內(比如UserDefaults或者Documents目錄),用一個字段保存老版本號
2、在開始初始化之前,讀取老版本號,以及當前版本號
3、如果該應用是第一次加載,那麼老版本號就取不到(因為是初次加載,這個字段還沒有保存),那麼就可以執行初始化過程;如果取到了老版本號,就不執行初始化
4、初始化完成之後,執行數據遷移。因為有老版本號和新版本號,所以可以通過對比,實現增量式的遷移
5、上述動作都完成之後,刷新老版本號
6、下次正常啟動,就不會再初始化,也不會執行數據遷移了;如果是安裝新版本,由於當前版本號刷新,又會觸發數據遷移
上面說的是比較簡單的場景。如果應用允許多用戶切換賬號,而且不同用戶的數據是分離的,就更復雜一些
首先標識老版本號的字段不能保存在UserDefaults裡,因為UserDefaults是用戶共享的。這樣當A用戶初始化之後,老版本號就存在了。切換到B用戶,發現老版本號已存在,則不會執行初始化,其實這時候B用戶的數據文件還沒有創建好。所以需要把老版本號存在單獨的地方,比如每個用戶各自的sqlite文件中
然後,讀取老版本號的時候,也要根據用戶的獨立標識去查詢
目前暫時是把老版本號保存在sqlite裡,但是這樣首次讀取的時候,判斷邏輯比較麻煩。需要判斷sqlite文件是否存在,然後要判斷table有沒有,最後才能取值。如果用文本保存可能會稍微方便一點,比存在sqlite裡,少了一個判斷table是否存在的步驟
-(BOOL) needInit { return [oldVersion isEqual: @"0"]; } -(NSString*) oldVersion { return oldVersion; } -(NSString*) currentVersion { return currentVersion; } #pragma mark - private method -(void) initOldVersion { // 數據庫文件不存在,oldVersion設為0 NSFileManager *fileManager = [NSFileManager defaultManager]; NSString *dbFilePath = [YLSGlobalUtils getDatabaseFilePath]; if(![fileManager fileExistsAtPath:dbFilePath]){ oldVersion = @"0"; return; } // 數據庫文件打開失敗,oldVersion設為0 FMDatabase *db = [FMDatabase databaseWithPath:dbFilePath]; if(![db open]){ oldVersion = @"0"; return; } // tb_clientstage表不存在,oldVersion設為0 int tableCount = 0; FMResultSet *rs = [db executeQuery:@"select count(*) as count from sqlite_master where type='table' and name='tb_clientstage'"]; if([rs next]){ tableCount = [rs intForColumn:@"count"]; } if(tableCount == 0){ oldVersion = @"0"; [db close]; return; } // 設置oldVersion rs = [db executeQuery:@"select * from tb_clientstage where id = '1' and tableno = '0'"]; if([rs next]){ oldVersion = [rs stringForColumn:@"version"]; }else{ oldVersion = @"0"; } [db close]; } -(void) initCurrentVersion { NSDictionary* infoDict =[[NSBundle mainBundle] infoDictionary]; NSString* versionNum =[infoDict objectForKey:@"CFBundleVersion"]; currentVersion = versionNum; }
clientInfo = [YLSClientInfo new]; if([clientInfo needInit]){ [self createEverythingForFirstTime];// 初始化時才執行 } [self allTheTime];// 任何時候都執行 [migrationHelper doMigration:clientInfo];
-(void) doMigration:(YLSClientInfo*)clientInfo { NSString *oldVersion = [clientInfo oldVersion]; NSString *currentVersion = [clientInfo currentVersion]; // 正常登陸,不需要數據遷移 if([oldVersion isEqualToString:currentVersion]){ return; } // 全新安裝,非升級,不需要數據遷移 if([oldVersion isEqualToString:@"0"]){ return; } // 以下均是版本升級,需要數據遷移 if([oldVersion isEqualToString:@"1.0.0"]){ [script1 doMigration]; [script2 doMigration]; [script3 doMigration]; [script4 doMigration]; return; } // 其他的情況 }