6

我已经在那个上面花了几个小时,但我似乎找不到解决方案。首先,我有什么规格:

  • 带有 Core Data 的 Objective-C iOS 6 应用程序
  • Core Data 是从 UIManagedDocument 初始化的,它开启了自动保存
  • UIManagedDocument 中的 ManagedContext 存储在静态变量中,并在整个应用程序中重用
  • 该应用程序使用 RestKit,我使用它提供的 ActiveRecord 类别。

我有一个带有团队实体的模型。在应用程序中,有一个团队列表视图控制器,它从后端加载团队。后端返回 JSON 数组,其中包含团队 ID。我将此 ID 存储在模型的“id”字段中,并在迭代服务器响应时,查找给定 ID 的团队是否已经存在。如果是,我只更新它的信息并传递对象,如果不是 - 我先创建它。

这就是它变得奇怪的地方。这在 90% 的时间内都有效。我可以加载团队列表控制器,在应用程序中走得更远,回到控制器(再次加载数据),大多数时候一切都很好。但是,偶尔,我的 fetch 请求不会返回任何内容。如:

NSArray *results = [context executeFetchRequest:request error:&error];

将返回空数组,错误也是空的。服务器上的任何东西都不会改变。当我尝试调试它时,请求如下所示:

<NSFetchRequest: 0x127a0e20> (entity: Team; predicate: (id == "123"); sortDescriptors: ((null)); limit: 1; type: NSManagedObjectResultType; )

当我在外部应用程序中预览 sqlite 商店时,我可以看到该 ID 的 Team 项目存在,并且在其他 90% 的时间里它确实被加载了。更奇怪的是,当我打开 SQLDebug 时,我可以看到:

2013-04-12 12:00:27.934 SportLink[10831:c07] CoreData: annotation: fetch using
NSSQLiteStatement <0x12656d10> on entity 'Team' with sql text 'SELECT 0, t0.Z_PK,
t0.Z_OPT, t0.ZACI1, t0.ZACI2, t0.ZACI3, t0.ZALTFIRSTNAME, t0.ZALTSECONDNAME, t0.ZCOLOR1,
t0.ZCOLOR2, t0.ZGENDER, t0.ZICON, t0.ZID, t0.ZLOCATION, t0.ZLOGO1, t0.ZLOGO2, t0.ZNAME,
t0.ZTYPE, t0.ZSPORT FROM ZTEAM t0 WHERE  t0.ZID = ?  LIMIT 133' returned 6 rows

这意味着后备存储实际上获取了数据,但由于某种原因没有传递它。现在:

  • 我在主线程中做所有事情,所以这不是问题
  • 我相信自动保存也不是问题,因为即使我关闭应用程序并在已经存在数据的情况下再次打开它时也会发生这种情况。

关于这里到底发生了什么的任何想法?

编辑:我已经使用 Instruments 分析了该应用程序。我刚刚进入有问题的 VC,然后进入它的子控制器并离开,重复了几次,直到问题开始显现。这是结果。也许从这里知道发生了什么会更容易:

仪器输出

4

1 回答 1

5

经过几天的调查,我终于找到了这个bug的原因。获取 Team 实体的代码如下所示:

Team *team = [Team findFirstByAttribute:WSK_ID withValue:data[WSK_ID] inContext:moc];

WSK_ID只是一个定义@"id"data是我从 RestKit 获得的字典,来自 JSON 响应。因为它没有被强制转换为任何类型,所以它可能是一个 NSString,而idTeam 实体的属性是 NSNumber。虽然我不知道它有时会失败,但将此位更改为:

Team *team = [Team findFirstByAttribute:WSK_ID withValue:@([data[WSK_ID] intValue]) inContext:moc];

即将 ID 字符串转换为 int,然后用 NSNumber 装箱解决了这个问题。

于 2013-04-15T11:14:59.270 回答