6

此代码引发“CoreData:错误:(19)PRIMARY KEY 必须是唯一的”错误。该Day实体只有一个when属性,即一个NSDate和一个称为 的一对多关系tasks。为什么会出现这个错误?如果Day已经存储了具有特定日期的 a,则获取它,否则我将其插入。因此,对于每个 day 对象,都应该有不同的when属性。我不确定这是否是主键。如何解决这个问题?先感谢您。

   NSMutableSet *occurrences = nil;

   occurrences = ... 
   NSMutableOrderedSet *newSet = [NSMutableOrderedSet orderedSetWithCapacity:[occurrences count]];

   for(NSDate *current in occurrences) {

        NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

        // try to find a corresponding Day entity whose when attribute is equal to the current occurrence
        // if none is available, create it

        Day * day = [[self getDayForDate:current inManagedObjectContext:moc] retain];
        if(!day){
            day = (Day *) [NSEntityDescription insertNewObjectForEntityForName:@"Day" inManagedObjectContext:moc];
        }

        day.when = current;
        [day addTasksObject:aTask];

        [newSet addObject:day];

        [moc insertObject:day];
        [moc processPendingChanges];

        [day release];

        [pool release];            
    } 

- (Day *)getDayForDate:(NSDate *)aDate inManagedObjectContext:(NSManagedObjectContext *)moc
{        
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Day" inManagedObjectContext:moc];
    [request setEntity:entity];
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"(when == %@)", aDate];
    [request setPredicate:predicate];
    NSError *error = nil;
    NSArray *array = [moc executeFetchRequest:request error:&error];
    [request release];

    Day *theDay = nil;

    if(array && [array count] == 1){
        theDay = [array objectAtIndex:0];
    }

    return theDay;
}
4

2 回答 2

19

我最近一直在努力解决CoreData: error: (19) PRIMARY KEY must be unique内部 iOS 应用程序的错误,幸运的是经过一两天的研究和代码修改后发现了解决方案,我想在这里分享我的发现,希望它们对其他。

首先,一点背景

我们的内部应用程序最初从未使用任何代码来更新其 CoreData 存储 - 相反,当需要在应用程序中显示新数据时,每次应用程序构建迭代时,CoreData SQLite 后备存储文件被简单地替换为新版本已使用基于桌面的标准 SQLite 编辑器进行了编辑——即原始 SQLite 文件已根据需要进行修改和更新。虽然人们认识到这种方法没有考虑到 CoreData 的“黑盒”性质,但实际上在过去几年中,它在我们的简单应用程序中为我们提供了很好的服务,并且该应用程序很高兴地接受了更新后的 SQLite 文件。新的应用程序构建。

然而,最近我们对应用程序进行了重大改造,从通过 Xcode 和应用程序重建来更新应用程序资产和数据的模型转变为通过 Web 服务获取新应用程序数据的模型,而正是在这个新的开发过程中PRIMARY KEY must be unique问题出现的工作。

经过几天的努力解决问题,认为新的CoreData实体创建代码(已经检查和重新检查)或其他相关问题一定有问题,通过进一步研究发现我能够使用 Xcode 为我们的应用程序启用 CoreData SQL 调试(按照这个非常有用的 SO 帖子中的说明)。

当我仔细研究 Xcode 中的 SQL 日志时,我可以看到在 CoreData 每次调用INSERTSQLite 后备存储中的新记录之前,框架Z_PRIMARYKEY使用以下查询查询表,SELECT Z_MAX FROM Z_PRIMARYKEY WHERE Z_ENT = ?其中?在幕后用相关Z_ENT值替换对于相关的 CoreData 实体(您可以Z_ENT通过查看表的内容来查看每个 CoreData 实体的值Z_PRIMARYKEY)。这就是我终于明白 CoreData 的黑匣子里发生了什么的时候!然后,我在 Mac 上使用 Liya 应用程序仔细查看了我们应用程序的 SQLite 文件,并查看了Z_PRIMARYKEY表的内容,果然Z_MAX列值都设置为0. 当 CoreData 首次为我们的应用程序生成空的 SQLite 后备存储文件时,它们的初始值从未改变过!

我立即意识到出了什么问题,以及为什么 CoreData 报告了主键错误——它实际上与最初怀疑的任何更高级别的 CoreData 实体对象的属性冲突无关,但确实是一个较低级别的错误。直到现在这个错误才真正有意义,它一直存在,只是没有在正确的上下文中被理解。

进一步研究表明,我们的内部团队在过去几年中成功地直接编辑了 SQLite 后备存储,从各种实体表中插入、更新和删除记录,但我们的团队从未进行任何更改到Z_PRIMARYKEY桌子上,由于我们的应用程序一直以只读方式使用 CoreData,这从来都不是问题。

但是,现在我们的应用程序正在尝试创建 CoreData 条目并将其保存回 SQLite 后备存储,INSERT因此生成的许多查询都失败了,因为 CoreData 无法获得正确的最大主键值来生成任何给定表上的下一个顺序主键,因此INSERT查询将失败并出现现在可以理解的有意义的PRIMARY KEY must be unique错误!

解决错误

我意识到每个开发人员可能会以各种方式为他们的应用程序生成默认/初始 CoreData 内容,但是,如果您曾经在 CoreData 中遇到过这个特定错误并且也很难立即找到明确的答案,我希望以下想法有帮助:

  • 我们的应用程序的 CoreData 后备存储是通过直接编辑 SQLite 文件(从实体表中插入、更新和删除记录)手动更新的——这种方法已经工作了很长时间,因为我们一直在以只读方式使用这些数据我们的应用程序。然而,这种方法未能真正认识到 CoreData 的抽象“黑盒”性质以及 SQLite 记录不等同于 CoreData 实体,SQLite 连接不等同于 CoreData 关系等事实。

  • 如果您的应用程序使用任何类型的预填充 CoreData 存储,并且如果您以任何方式而不是通过 CoreData 填充 SQLite 后备存储的内容 - 请确保如果您确实在任何 CoreData 实体表中创建新记录,您还确保更新表中的相关记录Z_PRIMARYKEY,将列值设置为相关实体表中Z_MAX的当前最大值。Z_PK

    例如,如果您有一个名为 的 CoreData 实体, 则Employee在您的 CoreData SQLite 持久存储支持文件中,这将由一个名为的表表示ZEMPLOYEE-除了代表您实体的属性和关系。此实体表还将在表中具有相应的记录,其值为- 因此,当您直接向表中添加新记录时 - 确保在完成添加记录后,您遍历每个实体表并复制最大值值到表的列。您输入的值应该是最大的Z_PKZ_ENTZ_OPTZ_PRIMARYKEYZ_NAMEEmployeeZEMPLOYEEZ_PKZ_MAXZ_PRIMARYKEYZ_MAXZ_PK对应表中的值;不要设置Z_MAX为等于Z_PK+ 1 的值,因为这不是 CoreData 所期望的!

您可以Z_PK使用以下 SQL 查询任何实体表的最大值:

"SELECT Z_PK FROM ZEMPLOYEE ORDER BY Z_PK DESC LIMIT 1"

这将为您提供表Z_PK中任何记录的最大值ZEMPLOYEE- 显然您应该替换ZEMPLOYEE为应用程序数据模型的相关表名称。

对于我们的应用程序,我能够编写一个简单的命令行脚本直接读取 SQLite 文件,遍历Z_PRIMARYKEY表的每条记录并使用正确的Z_MAX值更新它。一旦完成,我就可以使用这个更新的文件作为应用程序的持久存储支持文件。现在,当我插入并保存新的 CoreData 实体时,一切都按预期工作。

现在我们的应用程序可以通过 Web 服务直接请求新数据,我们将完全摆脱手动编辑 SQLite 后备存储并专门使用 CoreData 框架来更新数据,但有时或其他应用程序,其中可能需要这种快速简便的数据输入,我希望这个答案可以帮助其他开发人员,并为他们节省我发现这个解决方案所花费的时间和精力。

于 2013-02-04T22:38:43.653 回答
7

我想如果你已经有了它,你不需要插入一个新Day的(当 day 不是 nil 时就是这种情况)。我特别指的是[moc insertObject:day].

如果您使用insertNewObjectForEntityForName,该方法会在您保存 moc 时为您插入对象。如果您需要修改它(您已检索到非零日),请修改并保存。此外,我会processPendingChanges在循环结束时执行(仅出于性能原因)。

希望有帮助。

于 2012-09-19T12:17:46.667 回答