我正在开发一个Cocoa应用程序,该应用程序使用FMDB与本地SQLite数据库进行通信。我遇到了一个问题,我无法对DB进行任何插入或更新操作。选择查询运行完全正常,所以我假设我的数据库连接设置是正确的。
我的代码结构基本上是这样的:
FMDatabase* db=[FMDatabase databaseWithPath:[[NSBundle mainBundle] pathForResource:@"DBName" ofType:@"sqlite"]];
if(![db open])
{
NSLog(@"Could not open db.");
}
db.traceExecution=YES;
[db beginTransation];
[db ExecuteUpdate:"INSERT INTO test (title) VALUES(?)", [NSNumber numberWithInt]:2],nil];
[db commit];
[db close];
执行期间没有抛出任何异常或警告,关于db.traceExecution的控制台输出如下:
<FMDatabase: 0x100511fd0> executeUpdate: BEGIN EXCLUSIVE TRANSACTION;
<FMDatabase: 0x100511fd0> executeUpdate: INSERT INTO test (title) VALUES(?);
obj: 2
<FMDatabase: 0x100511fd0> executeUpdate: COMMIT TRANSACTION;
测试数据库只是INT类型的一列表。
除了db文件根本没有更新外,一切看起来都很好。这对我来说真的很困惑,因为Select查询工作得很好。我检查了数据库的路径,它指向正确的路径。首先我怀疑它是由文件许可引起的,但即使我允许每个人都能够读/写,问题仍然存在。
我已经被这个问题困扰了好几个小时,找不到合适的解决方案。任何人都可以对此有所了解吗?谢谢!
答案 0 :(得分:1)
捆绑包中的数据库是只读的。如果文件不存在于您定义的目标文件夹中,则应将其从捆绑包复制到库或文档文件夹,然后连接到该文件夹。这意味着它将在首次使用该路径时复制。
这是一个通过将数据库从包中复制到目标来“准备”数据库的功能。它将它复制到库(从我的iOS应用程序),但您可以复制到任何你想要的地方。在我的例子中,它是contacts.db。
我从ensureOpened调用了这个方法。
- (BOOL)ensureDatabasePrepared: (NSError **)error
{
// already prepared
if ((_dbPath != nil) &&
([[NSFileManager defaultManager] fileExistsAtPath:_dbPath]))
{
return YES;
}
// db in main bundle - cant edit. copy to library if !exist
NSString *dbTemplatePath = [[NSBundle mainBundle] pathForResource:@"contacts" ofType:@"db"];
NSLog(@"%@", dbTemplatePath);
NSString *libraryPath = [NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSUserDomainMask, YES) lastObject];
_dbPath = [libraryPath stringByAppendingPathComponent:@"contacts.db"];
NSLog(@"dbPath: %@", _dbPath);
// copy db from template to library
if (![[NSFileManager defaultManager] fileExistsAtPath:_dbPath])
{
NSLog(@"db not exists");
NSError *error = nil;
if (![[NSFileManager defaultManager] copyItemAtPath:dbTemplatePath toPath:_dbPath error:&error])
{
return NO;
}
NSLog(@"copied");
}
return YES;
}