对不起,很长的帖子;但是,我想通过。
我有多个托管对象上下文(moc)和一个稳定且一致的大量代码。除了当前的问题。奇怪的是moc的地址不是我创建的地址 - 一个无关的moc崩溃,我甚至没有访问。如果你有时间,请继续阅读。
非常简单地说,我创建一个新的子moc并将其传递给新的视图控制器。用户设置一些值并保存它们。保存子moc,然后保存父。但是,在编辑视图控制器中,如果用户选择显示Google Place Picker,就像开始显示窗口一样,我会收到多线程错误。我在iPhone 6 Plus实际设备上出现此错误,但在iPhone 6S上没有。它没有任何问题。使用6 Plus时,总是在同一个地方崩溃。
我已启用com.apple.CoreData.ConcurrencyDebug 1
这是创建子视图以进行编辑的主要方法:
// =========================================================================
// EventsTableViewController
class EventsTableViewController: UITableViewController {
// =========================================================================
// MARK: - Target Action Methods
@IBAction func didTapNewEvent(sender: UIBarButtonItem) {
guard let vc = storyboard?.instantiateViewControllerWithIdentifier(EditEventTableViewController.identifier) as? EditEventTableViewController else { DLog("ERROR creating EditEventTableViewController"); return }
// create a child context
let eventContext = NSManagedObjectContext(concurrencyType: .MainQueueConcurrencyType)
eventContext.parentContext = context
eventContext.name = "child_of_mainContext_for_Event"
let event = NSEntityDescription.insertNewObjectForEntityForName(Event.entityName, inManagedObjectContext: eventContext)
vc.segueObject = event
vc.context = eventContext
vc.shouldDismiss = true
vc.delegate = self
let nc = UINavigationController(rootViewController: vc)
presentViewController(nc, animated: true, completion: nil)
}
}
这是声明属性的编辑视图控制器:
// =========================================================================
// EditEventTableViewController
class EditEventTableViewController: UITableViewController {
// =========================================================================
// MARK: - Google Place properties
private var context: NSManagedObjectContext!
private var placePicker: GMSPlacePicker!
}
这是我调用show map函数的地方:
// =========================================================================
// MARK: - UITableViewDelegate
extension EditEventTableViewController {
override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
if indexPath == Cell.Location.indexPath {
didTapSelectFromMap()
}
}
}
这是谷歌的地方选择器,每次代码崩溃完全相同的地点,我甚至没有访问任何上下文(我实际上,但删除它们并没有解决问题):
// =========================================================================
// MARK: - Google Places
extension EditEventTableViewController {
func didTapSelectFromMap() {
guard let event = selectedObject as? Event else { DLog("ERROR object is not Event"); return }
guard let location = locationManager.location else {
DLog("Cannot get location, will try again")
locationManager.startUpdatingLocation()
return
}
DLog("current location: \(location)")
// create viewport around where the user is
let center = CLLocationCoordinate2DMake(location.coordinate.latitude, location.coordinate.longitude)
let northEast = CLLocationCoordinate2DMake(center.latitude + 0.001, center.longitude + 0.001)
let southWest = CLLocationCoordinate2DMake(center.latitude - 0.001, center.longitude - 0.001)
let viewport = GMSCoordinateBounds(coordinate: northEast, coordinate: southWest)
let config = GMSPlacePickerConfig(viewport: viewport)
placePicker = GMSPlacePicker(config: config)
placePicker.pickPlaceWithCallback { // the code crashes here! But I have no idea where the violation occurs.
(place, error) in
}
}
}
我将所有内容写入日志,日志中的相关信息如下。正如您所看到的,我创建了4个上下文,显示了地址,但错误是在我尚未创建的某个MOC上。可能是这样,GooglePlacePicker正在使用它自己的moc,并以某种方式与我的混合:)??? ???
[00059]:CDStack.swift :parentContext................. :12:41:18 CREATED <NSManagedObjectContext: 0x1741dbe40>: parent
[00068]:CDStack.swift :context....................... :12:41:18 CREATED <NSManagedObjectContext: 0x1741dbd50>: main
[00077]:CDStack.swift :importContext................. :12:41:18 CREATED <NSManagedObjectContext: 0x1701dd3d0>: import
[00095]:CDStack.swift :firebaseContext............... :12:41:21 CREATED <NSManagedObjectContext: 0x1741dc020>: firebase
[00127]:EditEventTableViewController.s :viewDidLoad()................. :12:43:48 Context: <NSManagedObjectContext: 0x1741de3c0>: child_of_mainContext_for_Event
[00375]:EditEventTableViewController.s :didTapSelectFromMap()......... :12:43:54 current location: <+78.675603,-93.352320> +/- 1414.00m (speed -1.00 mps / course -1.00) @ 25/09/2016, 12:43:54 Southern European Spring Time
2016-09-25 12:43:54.630499 App[1504:413029] [error] error: The current thread is not the recognized owner of this NSManagedObjectContext(0x1703c3de0). Illegal access during objectRegisteredForID:
我正在使用Xcode版本8.1 beta(8T29o),Swift 2.3但是在Xcode版本7.3.1中看到了相同的行为。
非常感谢任何指针。
休息时的堆栈如下所示。它不显示违规发生的行。如果确实如此,那么追踪这个错误会容易得多。
编辑1 - 保存功能
class func save(moc:NSManagedObjectContext) {
moc.performBlockAndWait {
if moc.hasChanges {
do {
try moc.save()
} catch {
DLog("ERROR saving context '\(moc)' - \(error)")
}
} else {
// there are no changes
}
if let parentContext = moc.parentContext {
save(parentContext)
}
}
}
答案 0 :(得分:0)
您正在-[NSManagedObjectContext save:]
完成区中调用-[GMSTileDataCache storeCacheableTileDatas:completionHandler:]
。这是在后台线程上执行的,我怀疑你正在调用与该后台线程无关的上下文保存。
快速回答是将保存包装在-performBlockAndWait:
。