为什么我在使用 NSManagedObject 时会在自身上发生崩溃清除基于块的 KVO?
Why do I get a crash clearing block-based KVO on self with NSManagedObject?
我有一个具有复杂关系的核心数据模型,我用中间 'Connection' 对象建模(此问题的进一步上下文:)。
我不希望使用模型的代码知道中间连接对象,所以我向我的主要托管对象添加了一个数组 属性,它公开了连接到的对象(我想要也可以观察到)。
@objc dynamic public internal(set) lazy var hosts: [Point] = {
let initialHosts = hostConnections.map { [=10=].superpoint }
hostsObservation = track(\Matter.hostConnections_!, on: self,
mapping: \HostConnection.superpoint, to: #keyPath(hosts))
return initialHosts
}()
上面的代码在 hosts
数组初始化时设置了对 NSOrderedSet
连接的观察(使用可以在我的所有连接类型上创建此模式的通用函数)。观察结果如期触发。
我使 willTurnIntoFault()
中的观察无效:
override public func willTurnIntoFault() {
hostsObservation?.invalidate()
print("hostsObservation \(hostsObservation) invalidated")
super.willTurnIntoFault()
}
但是,当这个对象在 NSKVODeallocate
中被释放时我遇到了崩溃。
hostsObservation Optional(<_NSKeyValueObservation: 0x600000ccd920>) invalidated
2019-11-04 12:02:06.467620+0000 Frame[27900:5614740] [General] An instance 0x60000300bc30 of class FrameGraph.Subject_Subject_ was deallocated while key value observers were still registered with it. Current observation info: <NSKeyValueObservationInfo 0x6000002d6ae0> (
<NSKeyValueObservance 0x600000ccd950: Observer: 0x600000ccd920, Key path: hostConnections_, Options: <New: NO, Old: NO, Prior: NO> Context: 0x0, Property: 0x600000c78b10>
...
0 CoreFoundation 0x00007fff3f590e45 __exceptionPreprocess + 256
1 libobjc.A.dylib 0x00007fff6a1e63c6 objc_exception_throw + 48
2 CoreFoundation 0x00007fff3f590c77 +[NSException raise:format:] + 193
3 Foundation 0x00007fff4180f349 NSKVODeallocate + 442
4 CoreData 0x00007fff3f060772 -[_PFManagedObjectReferenceQueue _processReferenceQueue:] + 1154
正如您从日志中看到的那样,我正在使崩溃前的观察无效,声称这是一个问题。如果我尝试将观察设置为 nil
willTurnToFault()
那时我的应用程序崩溃了。
我是 Core Data 的新手,任何人都可以帮助我解决我在这里缺少的问题吗?为什么观察它自己的对象 属性 以这种方式崩溃?
编辑 1:
我的跟踪功能是这样的:
internal func track<P: Point, C: Connection>(_ trackedPath: KeyPath<P, NSOrderedSet>,
on point: P,
mapping connectionPath: KeyPath<C, Point>,
to mappedPoints: String)
-> NSKeyValueObservation
{
return point.observe(trackedPath) { [unowned point] data, change in
// ...
}
}
编辑 2:
我已经将崩溃代码剥离回这个:
public class Matter: Point
{
override public func awakeFromFetch() {
super.awakeFromFetch()
print("\(hosts)") // Initialise lazy member and set observation
}
public override func willTurnIntoFault() {
super.willTurnIntoFault()
hostsObservation = nil // <-- EXC_BAD_ACCESS
}
@objc dynamic public internal(set) lazy var hosts: [Point] = {
hostsObservation = observe(\.hostConnections_) { [unowned self] data, change in } // Empty observation closure
return []
}()
private var hostsObservation: NSKeyValueObservation?
}
// Matter+CoreDataProperties.swift - auto generated
extension Matter {
@nonobjc public class func fetchRequest() -> NSFetchRequest<Matter> {
return NSFetchRequest<Matter>(entityName: "Matter")
}
@NSManaged public var hostConnections_: NSOrderedSet?
}
我不明白 hostsObservation
在 willTurnToFault()
崩溃时将其设置为 nil
会发生什么。
我正在构建 OSX 10.12,在 Xcode 11.1 中使用 Swift 5。
编辑 3:
在一个新的简单项目中转载:https://github.com/GilesHammond/KVO-Core-Data-Crash
关键问题似乎是我的核心数据模型在模块中。
我更新了示例项目以在此处显示潜在错误。
我已经与 Apple 支持人员交谈过。好像有系统bug。 Swift 当前无法正确清除模块中 NSManagedObject 上的 NSKeyValueObservation。
我已通过反馈助手向 Apple 提交问题。
我有一个具有复杂关系的核心数据模型,我用中间 'Connection' 对象建模(此问题的进一步上下文:
我不希望使用模型的代码知道中间连接对象,所以我向我的主要托管对象添加了一个数组 属性,它公开了连接到的对象(我想要也可以观察到)。
@objc dynamic public internal(set) lazy var hosts: [Point] = {
let initialHosts = hostConnections.map { [=10=].superpoint }
hostsObservation = track(\Matter.hostConnections_!, on: self,
mapping: \HostConnection.superpoint, to: #keyPath(hosts))
return initialHosts
}()
上面的代码在 hosts
数组初始化时设置了对 NSOrderedSet
连接的观察(使用可以在我的所有连接类型上创建此模式的通用函数)。观察结果如期触发。
我使 willTurnIntoFault()
中的观察无效:
override public func willTurnIntoFault() {
hostsObservation?.invalidate()
print("hostsObservation \(hostsObservation) invalidated")
super.willTurnIntoFault()
}
但是,当这个对象在 NSKVODeallocate
中被释放时我遇到了崩溃。
hostsObservation Optional(<_NSKeyValueObservation: 0x600000ccd920>) invalidated
2019-11-04 12:02:06.467620+0000 Frame[27900:5614740] [General] An instance 0x60000300bc30 of class FrameGraph.Subject_Subject_ was deallocated while key value observers were still registered with it. Current observation info: <NSKeyValueObservationInfo 0x6000002d6ae0> (
<NSKeyValueObservance 0x600000ccd950: Observer: 0x600000ccd920, Key path: hostConnections_, Options: <New: NO, Old: NO, Prior: NO> Context: 0x0, Property: 0x600000c78b10>
...
0 CoreFoundation 0x00007fff3f590e45 __exceptionPreprocess + 256
1 libobjc.A.dylib 0x00007fff6a1e63c6 objc_exception_throw + 48
2 CoreFoundation 0x00007fff3f590c77 +[NSException raise:format:] + 193
3 Foundation 0x00007fff4180f349 NSKVODeallocate + 442
4 CoreData 0x00007fff3f060772 -[_PFManagedObjectReferenceQueue _processReferenceQueue:] + 1154
正如您从日志中看到的那样,我正在使崩溃前的观察无效,声称这是一个问题。如果我尝试将观察设置为 nil
willTurnToFault()
那时我的应用程序崩溃了。
我是 Core Data 的新手,任何人都可以帮助我解决我在这里缺少的问题吗?为什么观察它自己的对象 属性 以这种方式崩溃?
编辑 1:
我的跟踪功能是这样的:
internal func track<P: Point, C: Connection>(_ trackedPath: KeyPath<P, NSOrderedSet>,
on point: P,
mapping connectionPath: KeyPath<C, Point>,
to mappedPoints: String)
-> NSKeyValueObservation
{
return point.observe(trackedPath) { [unowned point] data, change in
// ...
}
}
编辑 2:
我已经将崩溃代码剥离回这个:
public class Matter: Point
{
override public func awakeFromFetch() {
super.awakeFromFetch()
print("\(hosts)") // Initialise lazy member and set observation
}
public override func willTurnIntoFault() {
super.willTurnIntoFault()
hostsObservation = nil // <-- EXC_BAD_ACCESS
}
@objc dynamic public internal(set) lazy var hosts: [Point] = {
hostsObservation = observe(\.hostConnections_) { [unowned self] data, change in } // Empty observation closure
return []
}()
private var hostsObservation: NSKeyValueObservation?
}
// Matter+CoreDataProperties.swift - auto generated
extension Matter {
@nonobjc public class func fetchRequest() -> NSFetchRequest<Matter> {
return NSFetchRequest<Matter>(entityName: "Matter")
}
@NSManaged public var hostConnections_: NSOrderedSet?
}
我不明白 hostsObservation
在 willTurnToFault()
崩溃时将其设置为 nil
会发生什么。
我正在构建 OSX 10.12,在 Xcode 11.1 中使用 Swift 5。
编辑 3:
在一个新的简单项目中转载:https://github.com/GilesHammond/KVO-Core-Data-Crash
关键问题似乎是我的核心数据模型在模块中。
我更新了示例项目以在此处显示潜在错误。
我已经与 Apple 支持人员交谈过。好像有系统bug。 Swift 当前无法正确清除模块中 NSManagedObject 上的 NSKeyValueObservation。
我已通过反馈助手向 Apple 提交问题。