错误描述
我有一个应用程序,其中使用核心数据(由 SQLite 支持)和以下NSManagedObject
子类在本地存储了一些数据
import CoreData
@objc(ScoutingEventData)
class ScoutingEventData: NSManagedObject {
@NSManaged var id: String?
@NSManaged var type: String?
@NSManaged var entityId: String?
@NSManaged var oldStateJson: NSData?
@NSManaged var newStateJson: NSData?
@NSManaged var eventDate: NSDate?
func toInsertEvent() throws -> ScoutingEvent.Insert {
guard let id = id else { fatalError("events should have an event id") }
guard let data = newStateJson else { fatalError("insert event should have newStateJson stored") }
// If I uncomment this line, the error goes away.
// Somehow by ensuring that data never gets deallocated, the error never occurs.
// globallllll = data
return ScoutingEvent.Insert(id: id, entity: try ScoutingEntity.from(data))
}
}
// debugging var to prevent data from being deallocated
var globallllll: NSData?
正如上面片段中的注释中提到的,如果我允许newStateJson
释放从属性中读取的值,则会发生错误。
我收到的错误来自后台线程:
如果我在诊断工具中启用僵尸,我会得到这个
如果启用了僵尸,我还会在控制台中收到以下消息:
2016-11-18 16:26:13.773 ScoutingData_Example[51750:4716636] *** -[CFData release]: 消息发送到释放的实例 0x7f8c4eb10ae0
以及以下堆栈跟踪:
到目前为止我尝试过的
我尝试将NSManagedObjectContext
用于获取此数据的内容存储在静态变量中,以确保上下文永远不会被释放,但这没有任何效果。
我尝试将NSData?
属性转换为String?
属性,并将数据存储为 Base64 编码字符串而不是二进制数据(并且还更新了支持模型),但这也没有效果。错误仍然存在。
我试过注释掉读取属性的代码,错误消失了,但这显然不是一个可接受的解决方案。
我尝试在读取值后将值存储在全局变量中,以防止它被释放,并且错误消失了,但这也不是一个有效的解决方案。
我已经尝试注释掉所有使用该值的代码,而是只将值打印到控制台,并且错误仍然存在。这使我相信,完全访问该属性,然后再释放它的行为是为该错误创造条件的原因。
我很困惑。在我读取数据后如何处理数据似乎真的很奇怪,而且当后台线程被释放时发生任何事情似乎特别奇怪。
更奇怪的是,它似乎特定于这一属性。例如,我在读取 newDataJson 属性的行上方读取的 id 属性不会导致任何问题。您可能会认为这不同,因为 id 是一个字符串,而不是 NSData,但我尝试将我的 NSData 属性转换为字符串属性,但它仍然没有改变错误。
任何想法将不胜感激。谢谢。
编辑
不确定这是否有帮助,但这是我的模型架构