报错的位置在 Model 里面字段属性的 codegen 部分,具体来说大概是这样的位置:
{
@storageRestrictions(accesses: _$backingData, initializes: _child)
init(initialValue) {
_$backingData.setValue(forKey: \._child, to: initialValue)
_jobDescription = _SwiftDataNoType()
}
...
}
报错内容是「 illegal attempt to establish a relationship between objects in different contexts 」。
如果按照字面上的意思,也已经按照这里( https://stackoverflow.com/questions/77485748/json-to-swiftdata-decode-of-json-ends-in-error-at-observationregistrar )的说法在所有 Model 的 init 函数里加上了 context 那段代码了。
之前也遇到类似的情况,不过不在 init 而是在 getValue 里面。报错EXEC_BREAKPOINT Code=1
或者Illegal attempt to establish a relationship
,查了 stackoverflow 或者 Developer 社区的说法是可以通过给每个 Model 提供无参构造器,给每个 relationship 字段提供默认值来解决问题。也就是例如
let child = Child()
modelContext.insert(child)
let parent = Parent()
modelContext.insert(parent)
parent.child = child
这样先存入 context 再把 relationship 接上来的办法来解决。
然后在实现 JSON 导入导出的时候又遇到类似的问题。不过 JSONDecoder 是通过调用 Model 里面的构造器来创建对象的吧。也就是这个
required init(from decoder: Decoder) throws {
guard let context = ... else {
...
}
let container = try decoder.container(keyedBy: CodingKeys.self)
self.child = try? container.decode(Child.self, forKey: .child)
...
}
换句话说就是得先把 Child 构造出来然后赋值给self.child
字段,而这就会触发Illegal attempt to establish a relationship 'child' between objects in different contexts
。
但是因为 JSON 的 Decode 过程,没法通过先构造一个无参数的 Parent 再把 child 加到 parent 上面的方式来绕过。
这种情况下该怎么解决 relationship 的问题呢?
尝试了加@Relationship
或者不加或者加不同的注解,也试过在 relationship 两边的 class 分别加上对面的引用或者只加一边,但每次都会在同样的位置报错。
另外也在模拟器里面发现有时候虽然报错但是程序退出重启之后会多出来那么一两个原 JSON 里面的数据,也就是说存在解析成功了的情况,所以感觉也很迷惑。