CoreData: 错误: (14) 数据库 I/O 错误

Posted

技术标签:

【中文标题】CoreData: 错误: (14) 数据库 I/O 错误【英文标题】:CoreData: error: (14) I/O error for database 【发布时间】:2013-09-12 20:08:50 【问题描述】:

在 XCode 中使用 Core Data 编译和运行项目时,我遇到了一个以前从未见过的错误:

 2013-09-12 16:59:10.156 myapp[57811:70b] CoreData: error: 
      (14) I/O error for database at /Users/administrador/Library/
         Application Support/iPhone Simulator/7.0/Applications/
         6BA67336-B093-46CF-8B11-E3595409DAC2/myapp.app/database.sqlite.  

         SQLite error code:14, 'unable to open database file'

生成此消息的代码是:

    psc = [[NSPersistentStoreCoordinator alloc]
                   initWithManagedObjectModel:self.managedObjectModel];
    NSURL *storeURL = [[NSBundle mainBundle] 
                         URLForResource:@"database" withExtension:@"sqlite"];
    [psc addPersistentStoreWithType:NSSQLiteStoreType 
             configuration:nil URL:storeURL 
             options:@NSReadOnlyPersistentStoreOption : @YES error:NULL];

我试过Build->Clean,删除派生数据,卸载应用。

我在发帖前检查了this question,我相信问题有所不同。

注意: sqlite 是应用的资源

使用调试建议的信息

2013-09-12 17:43:38.341 myapp[58322:70b] CoreData: annotation: Connecting to sqlite database file at "/Users/administrador/Library/Application Support/iPhone Simulator/7.0/Applications/6BA67336-B093-46CF-8B11-E3595409DAC2/myapp.app/database.sqlite"
2013-09-12 17:43:38.360 myapp[58322:70b] CoreData: sql: SELECT Z_VERSION, Z_UUID, Z_PLIST FROM Z_METADATA
2013-09-12 17:43:38.363 myapp[58322:70b] CoreData: annotation: Disconnecting from sqlite database due to an error.
2013-09-12 17:43:38.364 myapp[58322:70b] CoreData: error: (14) I/O error for database at /Users/administrador/Library/Application Support/iPhone Simulator/7.0/Applications/6BA67336-B093-46CF-8B11-E3595409DAC2/myapp.app/database.sqlite.  SQLite error code:14, 'unable to open database file'
2013-09-12 17:43:38.366 myapp[58322:70b] CoreData: annotation: Disconnecting from sqlite database.

【问题讨论】:

【参考方案1】:

现在 ios7 上的 NDA 已经解除,我可以发布我为这个问题找到的解决方法。

iOS7中的Core Data默认使用WAL中的sqlite

唯一有效的解决方案是使用没有WAL的iOS6模拟器创建sqlite并将其导入项目中:

- (NSPersistentStoreCoordinator *)persistentStoreCoordinator

    static NSPersistentStoreCoordinator *psc;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^
        psc = [[NSPersistentStoreCoordinator alloc] 
                 initWithManagedObjectModel:self.managedObjectModel];
        NSURL *storeURL = [[NSBundle mainBundle] 
                  URLForResource:@"database" withExtension:@"sqlite"];
        [psc addPersistentStoreWithType:NSSQLiteStoreType
                          configuration:nil
                                    URL:storeURL
                                options:@NSReadOnlyPersistentStoreOption : @YES,
                             NSSQLitePragmasOption: @@"journal_mode":@"DELETE"
                                  error:NULL];
    );
    return psc;

【讨论】:

这里更容易禁用 WAL:***.com/a/18870738/288860 非常感谢,一直在网上寻找解决方案。令人沮丧的是,在 IOS 中使用预填充只读数据库的文档是如此缺乏。【参考方案2】:
NSURL *storeURL = [[NSBundle mainBundle] 
                     URLForResource:@"database" withExtension:@"sqlite"];

在应用程序包内构建一个只读路径。持久存储 文件需要驻留在可写目录中,例如“文档”目录。

编辑:上述答案实际上是错误的,可以从应用程序包中打开一个只读核心数据文件(使用NSReadOnlyPersistentStoreOption)。我目前唯一能想到的是 捆绑的文件不是有效的核心数据数据库。添加启动参数

-com.apple.CoreData.SQLDebug 1

可能有助于定位问题。

【讨论】:

虽然您可能是对的,但它无法从只读位置打开只读数据库不寻常的。 它总是有效的。我不确定为什么只读数据库必须驻留在 Documents 中 @ppaulojr:你是对的(我写这个答案太快了 :-) - 捆绑的文件可能不是有效的 Core Data 数据库(与当前模型匹配)吗? @ppaulojr:也许添加启动参数“-com.apple.CoreData.SQLDebug 1”以获取更多信息。 我添加了来自SQLDebug的信息

以上是关于CoreData: 错误: (14) 数据库 I/O 错误的主要内容,如果未能解决你的问题,请参考以下文章

从 Coredata 中删除一个对象(出现错误)

Coredata中的孩子数

调试 CoreData 异常

从实体中删除属性后迁移 CoreData 时出错

CoreData TextField - 意外发现 nil

为啥 deleteObject: 导致 CoreData 故障?