如标题中所述,在后台线程上保存对象后,其属性在主线程上为空,例如字符串为""、数字为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.asyncobjectID)在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从主上下文中提取该用户。现在您可以查看它是如何填充的。

08-04 13:58