如标题中所述,在后台线程上保存对象后,其属性在主线程上为空,例如字符串为""
、数字为0
等。
这是一些密码!
用户类:
@objc(User)
class User: NSManagedObject {
@NSManaged var id: Int32
@NSManaged var email: String
}
实际保存发生的用户存储库:
func saveUser(fromJSON json: Any, onSuccess: ((User) -> Void)?, onFailure: ((Error) -> Void)?) {
dataManager.persistentContainer.performBackgroundTask { context in
context.mergePolicy = NSMergePolicy.mergeByPropertyObjectTrump
let user = self.userFactory.user(fromJSON: json, inContext: context)
// print(user.id) and print(user.email) output correct data
do {
try context.save()
DispatchQueue.main.async {
// print(user.id) and print(user.email) show 0 and ""
onSuccess?(user)
}
} catch let error {
DispatchQueue.main.async {
onFailure?(error)
}
}
}
}
其中
dataManager
配置如下:class CoreDataManager {
private let modelName: String
lazy var persistentContainer: NSPersistentContainer = {
let container = NSPersistentContainer(name: modelName)
container.loadPersistentStores(completionHandler: { (storeDescription, error) in
if let error = error as NSError? {
fatalError("Unresolved error \(error), \(error.userInfo)")
}
container.viewContext.automaticallyMergesChangesFromParent = true
container.viewContext.mergePolicy = NSMergePolicy.mergeByPropertyObjectTrump
})
return container
}()
init(modelName: String) {
self.modelName = modelName
}
}
我想
userFactory
不是那么重要,但为了完整起见:class UserFactory {
func user(fromJSON json: Any, inContext context: NSManagedObjectContext) -> User {
guard
let jsonDictionary = json as? [String: Any?],
let userDictionary = jsonDictionary["user"] as? [String: Any?],
let id = userDictionary["id"] as? Int32,
let email = userDictionary["email"] as? String
else {
fatalError("Could not parse User object.")
}
let user = User(context: context)
user.id = id
user.email = email
return user
}
}
请参阅上面关于print语句的片段中的注释,其中指出了用户属性很好的地方,以及它们显示
0
和""
的地方。我知道我不能在线程之间传递nsmanagedObjects,但即使我尝试使用其
DispatchQueue.main.async
(objectID
)在NSManagedObjectID
块中获取用户对象,用户属性仍然为空。我想我做错了什么事,但一根手指也插不上。任何建议都将不胜感激。
非常感谢!
最佳答案
我认为问题在于核心数据不是线程安全的。我知道你知道这一点——毕竟,这就是为什么你创造了一个特殊的背景——但它有着非常严格的含义。你是说:
dataManager.persistentContainer.performBackgroundTask { context in
context.mergePolicy = NSMergePolicy.mergeByPropertyObjectTrump
let user = self.userFactory.user(fromJSON: json, inContext: context)
do {
try context.save()
DispatchQueue.main.async {
print(user) // <--
你不能那样做。您仍在谈论保存到后台上下文中的
user
,但您已经切换了线程。不,您不能从两个不同的线程谈论这个问题。一旦进入了主线程,与核心数据交谈的唯一方式就是通过主上下文。您需要根据用户的ID从主上下文中提取该用户。现在您可以查看它是如何填充的。