领域更新数据+ DispatchQueue

时间:2018-05-16 18:14:46

标签: ios swift realm

我在后台更新了DataBase。我的数据可以包含大约2000个项目,需要时间来更新。

func updateData(items: [JSON], _ complete:@escaping() -> Void) {
    DispatchQueue.global(qos: .userInitiated).async {

       let currentModels = EgrnModel.getAllModels()
       var newModels: [EgrnModel] = []
       var toDelete: [EgrnModel] = []

       for model in currentModels {
           let contain = items.contains(where: {$0["id"].intValue == model.id})
           if !contain {
               toDelete.append(model)
           }
       }


       let realm = try! Realm()
       try! realm.write {
           for item in items {
               if let model = currentModels.first(where: {$0.id == item["id"].intValue}) {
                   model.update(item)
               }
               else {
                   newModels.append(EgrnModel(item))
               }
           }
           realm.delete(toDelete)
           realm.add(newModels)
       }

       DispatchQueue.main.async {
           complete()
       }
    }
}

我有一个功能,我需要暂时更新数据。当我点击复选标记时,我冻结了。 (我认为这是因为此时其他数据正在后台更新)

func checkMark(index: Int) {

    let model = models[index]
    let realm = try! Realm()

    try! realm.write {
        model.needToUpdateOnServer = true
        model.lastEditUpdate = Date()
        model.read = true
    }
}

我尝试下一个代码来修复冻结。但是在这段代码中,我遇到了崩溃Terminating app due to uncaught exception 'RLMException', reason: 'Realm accessed from incorrect thread.

func checkMark(index: Int) {

    let model = models[index]

    DispatchQueue.global(qos: .userInitiated).async {

        let realm = try! Realm()

        try! realm.write {
            model.needToUpdateOnServer = true
            model.lastEditUpdate = Date()
            model.read = true
        }
    }
}

1 个答案:

答案 0 :(得分:0)

您需要做的是将领域对象从一个线程“移动”到另一个线程,因为领域对象不是线程安全的,而是线程受限的。为此,您必须使用@override void didUpdateWidget(TextField oldWidget) { super.didUpdateWidget(oldWidget); if (widget.controller == null && oldWidget.controller != null) _controller = TextEditingController.fromValue(oldWidget.controller.value); else if (widget.controller != null && oldWidget.controller == null) _controller = null; // ... } API。

要解决此问题,请执行以下操作:

  1. 在领域类上创建扩展
ThreadSafeReference
  1. 以这种方式在您的代码中使用它
extension Realm {
    func writeAsync<T : ThreadConfined>(obj: T, errorHandler: @escaping ((_ error : Swift.Error) -> Void) = { _ in return }, block: @escaping ((Realm, T?) -> Void)) {
        let wrappedObj = ThreadSafeReference(to: obj)
        let config = self.configuration
        DispatchQueue(label: "background").async {
            autoreleasepool {
                do {
                    let realm = try Realm(configuration: config)
                    let obj = realm.resolve(wrappedObj)

                    try realm.write {
                        block(realm, obj)
                    }
                }
                catch {
                    errorHandler(error)
                }
            }
        }
    }
}

切换愉快!