gpt4 book ai didi

swift - 为什么我使用 NSManagedObject 会得到一个基于 block 的崩溃清除 KVO?

转载 作者:行者123 更新时间:2023-12-03 16:21:52 28 4
gpt4 key购买 nike

我有一个具有复杂关系的核心数据模型,我使用中间“连接”对象对其进行了建模(此问题的进一步上下文: Maintaining Complex Unidirectional Relationships in Core Data )。

我不希望使用模型的代码了解中间连接对象,因此我向主要托管对象添加一个数组属性,该属性公开连接到的对象(我也希望可以观察到)。

@objc dynamic public internal(set) lazy var hosts: [Point] = {
let initialHosts = hostConnections.map { $0.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

正如您从日志中看到的,我在崩溃之前使观察无效,声称这是一个问题。如果我尝试将观察设置为nilwillTurnToFault() 中,我的应用程序此时崩溃了。

我是核心数据新手,任何人都可以帮助我解决我在这里缺少的内容吗?为什么观察到自己属性的对象会以这种方式崩溃?

编辑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?

}

我不明白在 willTurnToFault() 崩溃时将 hostsObservation 设置为 nil 会发生什么情况。

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

编辑3:

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

最佳答案

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

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

我已与 Apple 支持人员进行了交谈。看来系统有bug。当 NSManagedObject 位于模块中时,Swift 目前无法正确清理 NSKeyValueObservation。

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

关于swift - 为什么我使用 NSManagedObject 会得到一个基于 block 的崩溃清除 KVO?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58693425/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com