为什么我在使用 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?

    }

我不明白 hostsObservationwillTurnToFault() 崩溃时将其设置为 nil 会发生什么。

我正在构建 OSX 10.12,在 Xcode 11.1 中使用 Swift 5。

编辑 3:

在一个新的简单项目中转载:https://github.com/GilesHammond/KVO-Core-Data-Crash

关键问题似乎是我的核心数据模型在模块中。

我更新了示例项目以在此处显示潜在错误。

我已经与 Apple 支持人员交谈过。好像有系统bug。 Swift 当前无法正确清除模块中 NSManagedObject 上的 NSKeyValueObservation。

我已通过反馈助手向 Apple 提交问题。