iOS SQLite 数据库迁移

作者: iYiming | 来源:发表于2015-04-20 21:14 被阅读11865次

最近不得不考虑关于数据库迁移的问题,原先用了种很不好的处理方式(每次版本升级就删除本地数据库,太傻),于是开始考虑下如何迁移数据库。

项目使用的 FMDB ,除了使用 Core Data 外,这就是最好的了(最近好像又有了个 realm )。

FMDB 介绍页面,发现了 FMDBMigrationManager ,大喜。

看了半天文档,捣鼓了半天才弄出来,一步步整理下。

0.安装 FMDBMigrationManager

Podfile 文件:

platform :ios, "7.0"

pod 'FMDB'
pod 'FMDBMigrationManager'

使用pod install命令安装

1.FMDBMigrationManager 创建数据库

FMDBMigrationManager *manager = [FMDBMigrationManager managerWithDatabaseAtPath:[YMDatabaseHelper databasePath]  migrationsBundle:[NSBundle mainBundle]];

其中[YMDatabaseHelper databasePath]是数据库路径

2.创建迁移表

BOOL resultState = [manager createMigrationsTable:&error];

创建的迁移表名称为:schema_migrations

3.创建 .sql 文件

该文件用来存储每次升级使用的 SQL 语句。

FMDBMigrationManager 建议我们使用时间戳来作为版本号,使用下面的命令生成一个文件:

touch "`ruby -e "puts Time.now.strftime('%Y%m%d%H%M%S%3N').to_i"`"_CreateMyAwesomeTable.sql

我生成的文件名为:20150420170044940_CreateMyAwesomeTable.sql,其中20150420170044940 为迁移的版本号标识。

我们在 20150420170044940_CreateMyAwesomeTable.sql文件中创建一个用户表,写入:

CREATE TABLE User(
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
name TEXT
);

4.迁移函数

FMDBMigrationManager *manager = [FMDBMigrationManager managerWithDatabaseAtPath:[YMDatabaseHelper databasePath]  migrationsBundle:[NSBundle mainBundle]];
    
BOOL resultState = NO;
NSError *error = nil;
if (!manager.hasMigrationsTable) {
    resultState = [manager createMigrationsTable:&error];
}
    
resultState = [manager migrateDatabaseToVersion:UINT64_MAX progress:nil error:&error];//迁移函数

NSLog(@"Has `schema_migrations` table?: %@", manager.hasMigrationsTable ? @"YES" : @"NO");
NSLog(@"Origin Version: %llu", manager.originVersion);
NSLog(@"Current version: %llu", manager.currentVersion);
NSLog(@"All migrations: %@", manager.migrations);
NSLog(@"Applied versions: %@", manager.appliedVersions);
NSLog(@"Pending versions: %@", manager.pendingVersions);

UINT64_MAX 表示把数据库迁移到最大的版本

运行项目,打印出如下内容:

2015-04-20 20:50:19.033 YMFMDatabase[12654:1326201] Has `schema_migrations` table?: YES
2015-04-20 20:50:19.036 YMFMDatabase[12654:1326201] Origin Version: 20150420170044940
2015-04-20 20:50:19.036 YMFMDatabase[12654:1326201] Current version: 20150420170044940
2015-04-20 20:50:19.037 YMFMDatabase[12654:1326201] All migrations: (
    "<FMDBFileMigration: 0x17003b4c0>"
)
2015-04-20 20:50:19.037 YMFMDatabase[12654:1326201] Applied versions: (
    20150420170044940
)
2015-04-20 20:50:19.038 YMFMDatabase[12654:1326201] Pending versions: (
)

用 iFunBox 查看下是不是创建了一个 User 表,里面含有 idname 字段。以及 FMDBMigrationManager 生成的 schema_migrations 表。

5.创建第二个 .sql 文件

先用上方命令:

touch "`ruby -e "puts Time.now.strftime('%Y%m%d%H%M%S%3N').to_i"`"_CreateMyAwesomeTable.sql```

生成,我生成的是:``20150420170557221_CreateMyAwesomeTable.sql``。

第二个 sql 文件,里面创建一个新表名字为 ``Grouping``,为原先的 ``User`` 表添加邮箱字段:``email``。.sql 文件修改如下:

CREATE TABLE Grouping(
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
name TEXT
);

ALTER TABLE User ADD email TEXT;


##6.创建第三个 .sql 文件

生成同上,文件内容是为 ``Grouping`` 表添加备注字段 ``remark``:

文件内容:  

ALTER TABLE Grouping ADD remark TEXT;


OK,直接运行项目,看看是不是创建了 ``Grouping`` 表,里面含有``id``,``name``,``remark``字段,以及 ``User`` 表里面是不是添加了 ``email`` 字段。


##7.遇到的问题

中间我自己做 Demo 时,试图删除表中的某列,比如删除 ``User`` 表中的 ``name`` 列,但是不能成功,Google 了下发现[答案](http://stackoverflow.com/questions/8442147/how-to-delete-or-add-column-in-sqlite)。

>SQLite supports a limited subset of ALTER TABLE. The ALTER TABLE command in SQLite allows the user to rename a table or to add a new column to an existing table. It is not possible to rename a column, remove a column, or add or remove constraints from a table.

解释下:就是说 ``SQLite`` 对 ``ALERT TABLE`` 命令受限制,``SQLite`` 中的 ``ALERT TABLE`` 命令只能允许用户重命名表或者添加新列,不能重命名列或者删除列或者删除约束。

相关文章

  • Android 数据库

    sqlite详解 sqlite升级,增加字段的语句 数据库框架对比和源码分析 数据库的优化 数据库数据迁移问题

  • iOS对象存储框架,含单元测试Demo

    iOS database framework based on sqlite3(基于sqlite3的iOS数据库存...

  • iOS SQLite 数据库迁移

    最近不得不考虑关于数据库迁移的问题,原先用了种很不好的处理方式(每次版本升级就删除本地数据库,太傻),于是开始考虑...

  • iOS SQLite 数据库迁移

    1、概述 随着应用的不断升级,原有的数据库结构不再适应新的功能,这时候,就需要对SQLite数据库的结构进行迁移升...

  • DBFlow:数据迁移(Migrations)

    数据迁移是如何工作的? 在sqlite中,所谓迁移是修改现有的数据库的设计结构,以适应新的数据结构。在sqlite...

  • iOS 开发之 SQLite 数据库使用

    iOS 常用 SQLite 数据库存储数据,工程中需要引入 sqlite3 库。SQLite 提供 C 接口以供调...

  • iOS数据存储之文件沙盒

    资源连接: iOS数据库存储之SQL语句; iOS数据库存储之SQLite3; iOS数据存储之NSCoding;...

  • iOS数据库基本使用

    基本介绍 iOS数据库使用的是SQLite,一款轻型的嵌入式关系数据库。安卓和iOS开发使用的都是SQLite数据...

  • SQLite数据库基础(完结)

    标签(空格分隔): iOS数据库 数据库总结 SQLite数据库 ios中的数据存储方式及其特点 Preferen...

  • iOS SQLite数据库基础

    标签(空格分隔): iOS数据库 数据库总结 SQLite数据库 ios中的数据存储方式及其特点 Preferen...

网友评论

  • 刘超_a594:你好 下面这个方法是在数据库目录下生成spl吗 不知道我为什么生成不了这个user的信息
    touch "`ruby -e "puts Time.now.strftime('%Y%m%d%H%M%S%3N').to_i"`"_CreateMyAwesomeTable.sql
    刘超_a594:楼主不在的吧
  • samuelandkevin:楼主,可以参考我的:iOS-FMDB+runtime封装,简单实用,省去复杂的sql语句.(https://github.com/samuelandkevin/PackaingFMDB
  • LD_左岸:请教下 FMDBMigrationManager 这个第三方库和FMDB这个第三方库有什么关系 是一个作者 还是 FMDBMigrationManager 是FMDB的一部分?
  • zdl:能加载本地的 .sql文件 更新数据库吗
  • zhangferry:楼主你好,我照你的方式建迁移表但是打印的current version 和 origin version为0,查看后台没有新表,是怎么回事啊
    CodingLady:@勇闯天涯茉莉花茶 另外一种是写一个类实现FMDBMigrating协议,然后在migrateDatabase:error:方法里写database的更新语句
    CodingLady:@勇闯天涯茉莉花茶 官方文档里面介绍了两种迁移方式,一种是楼主说的通过.sql文件进行,这个时候必须要建一个FMDBFileMigration对象(比如这个对象叫a),然后调用FMDBMigrationManager的addMigration方法,把a加进去才可以执行更新
    CodingLady:@勇闯天涯茉莉花茶 我看了代码,schema_migrations必须有一条版本号的记录,才会执行迁移

本文标题:iOS SQLite 数据库迁移

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