App升级时数据库的迁移更新

作者: _轻舟 | 来源:发表于2015-11-05 10:41 被阅读5284次

前一段时间在进行App升级的时候,由于一开始版本初期没有考虑完善,导致走了很多弯路,后来经过自己的一些思考,总结出了一些在app升级的时候,数据库内文件同步保持更新的经验,希望能给大家带来帮助.

总体方案及思路

流程图
  1. 在每一次运行程序的时候,判断是否存在数据库,如果不存在则直接创建数据库,若存在取出数据库版本号进行其他的处理.

  2. ,当用户第一次下载安装app的时候,第一次建立版本库,将我们的数据信息存入数据库中,同时保存一个当前版本号加一的字段到数据库中.
    那么问题来了,为什么我们需要将版本信息加一呢,这是为了以后进行版本判断的时候更加方便.
    还有一个问题,为什么我们将版本信息放入数据库而不使用UserDefaults快速存储呢?原因是你需要考虑到
    当你的app有不同的用户登录时,UserDefaults是所有数据共享的,你不能根据不同的用户来处理他的信息
    判段他的信息是否需要更新

  3. 当用户更新app的时候,会直接从数据库中取出上一次保存的版本字段,例如是2.0版本的时候,会直接从case2开始执行,修改完数据结构以后,再一次将版本字段存到数据库中.

  4. 所以每更新一次版本,如果数据结构信息有变动的时候,直接在后面加case语句即可.

下面是一些参考代码,使用FMDB库:


/**
 *  1.当没有数据库存在的时候,创建数据库 然后执行版本1的创建(创建版本表,创建信息表),保存1.0+1.0信息到版本库,数据库内版本号为2.0
    2.当有数据库存在的时候,执行判断版本库的版本信息,版本号赋值给dbVersion,然后进行信息的更新,同时更新版本号的更新
 */
-(instancetype)init
{
    //设置数据库版本为1
    int dbVersion = 1;
    if (self = [super init]) {
        //判断本地有没有数据库文件
        if (![self isExistDB]) {
            //不存在 初始化数据库
            [self createDB];
        }else
        {
            //如果存在,那么获取版本信息
            _dataBase = [[FMDatabase alloc] initWithPath:[self getDBPath]];
            NSString * currentVersion = [self getDBInfoValue];
            dbVersion = currentVersion.intValue;
        }
        
        switch (dbVersion) {   //判断版本信息
            case 1:
            {
                //说明用户第一次安装  1.0版本
                //创建版本表
                [self excuteLocalSql:createTB_info];
                //创建信息表
                [self excuteLocalSql:create_tusersql];
                //保存1.0+1.0信息到数据库   用于下一次判断版本号
                [self setDBInfoValueWithString:@"2.0"];
            }
            case 2:
            {
                //更新信息表
                [self excuteLocalSql:update_tusersql];
                //保存2.0+1.0到数据库
                [self setDBInfoValueWithString:@"3.0"];
            }
                case 3:
                [self excuteLocalSql:modify];
                [self setDBInfoValueWithString:@"4.0"];
            default:
                break;
        }
        
    }
    return self;
}```

使用的方法:

//使用的SQL语句
static NSString *createTB_info=@"create table if not exists t_info (version text)";
static NSString *create_tusersql = @"CREATE TABLE IF NOT EXISTS T_User (userid INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, username TEXT, usergender TEXT, usercreatetime date, userupdatetime date,age INTEGER,islogined INTEGER)";
static NSString * update_tusersql = @"alter table T_User add column useraddress TEXT";
static NSString *modify=@"alter table T_User add column password text not null default '111111'";

//判断是否存在数据库
-(BOOL)isExistDB
{
NSFileManager *fileManager = [NSFileManager defaultManager];
return [fileManager fileExistsAtPath:[self getDBPath]];
}

//得到路径
-(NSString *)getDBPath
{
return [NSString stringWithFormat:@"%@/Documents/user1.sqlite",NSHomeDirectory()];
}

pragma mark - 更新版本号

-(BOOL)updateVersionInfoWithString:(NSString *)version
{
NSString *sql = @"UPDATE t_info SET version = ?";
BOOL b = [_dataBase executeUpdate:sql,version];
return b;
}

pragma mark - 插入版本号

-(BOOL)insertVersionInfoWithString:(NSString *)version
{
NSString *sql = @"INSERT INTO t_info(version) VALUES(?)";
BOOL b = [_dataBase executeUpdate:sql,version];
return b;
}

pragma mark - 创建版本信息

  • (BOOL)setDBInfoValueWithString:(NSString *)string
    {
    if ([self getDBInfoValue]) {
    //更新版本
    [self updateVersionInfoWithString:string];
    }
    else
    {
    //插入版本
    [self insertVersionInfoWithString:string];
    }
    return YES;
    }

//得到版本信息

  • (NSString *)getDBInfoValue
    {
    NSString *sql = @"select version from t_info";
    //注意 需要先打开数据库
    [_dataBase open];
    FMResultSet *resultSet = [_dataBase executeQuery:sql];

    NSString * version = nil;
    while ([resultSet next]) {
    version = [resultSet stringForColumn:@"version"];
    }
    return version;
    }

pragma mark - 创建数据库中的表

  • (BOOL)excuteLocalSql:(NSString *)createTB_info
    {
    BOOL b = [_dataBase executeUpdate:createTB_info];
    NSLog(@"表的创建结果:%d",b);
    return b;
    }

pragma mark - 初始化数据库

  • (void)createDB
    {
    _dataBase = [[FMDatabase alloc] initWithPath:[self getDBPath]];
    if (_dataBase.open == YES) {
    NSLog(@"数据库成功创建!");
    }else
    {
    NSLog(@"创建失败!");
    }
    }```

如果觉得对自己有帮助的话,请点赞噢!
有什么疑问和问题可以直接私信我.

相关文章

网友评论

  • 简单的逗号:是建了一个表用于存储版本号么
  • LD_左岸:是不是每次更新版本都需要增加一个case语句
  • Auditore:思路是不是这样:通过判断版本,然后把本地数据库取出来copy它的全部字段,然后创建新的表包含最新字段,然后删除旧表
    _轻舟:@auditore :+1:
    Auditore:看了你的文章觉得思路很清晰 感谢
  • ismilesky:有实现的Demo吗??有个问题就是,我要是多个版本,用swith case来区分是不是太多的分支了,有什么解决办法吗
  • STDawn:Mark
  • 9f94d02340f1:谢啦,学习学习!
  • zvving:区分版本是一种方式,之前还碰到过通过每行纪录增加更新时间字段来区分,做到更细粒度的更新进度控制
    _轻舟:@曾铭 嗯嗯 get
    zvving:@_轻舟 每条数据都有一个更新时间,服务器端维护数据库的历史变化:每次更新,一个更新时间 N 条记录。(可以认为每个更新时间戳就是一个版本)。这样客户端根据最新更新时间请求服务器逐条更新。用户能看到时间过渡的更新过程。这适用于客户端不产生数据的情况。实际效果可以看看用药助手的数据库更新:smile:
    _轻舟:@曾铭 不太会用,求教:smiley:

本文标题:App升级时数据库的迁移更新

本文链接:https://www.haomeiwen.com/subject/fxfchttx.html