CoreData 并发问题
CoreData Concurrency issue
我在使用 private managedObjectContext
在后台保存数据时遇到问题。我是 CoreData 的新手。我正在为 NSManagedObjectContext
使用父子方法,但面临几个问题。
多次点击刷新按钮出现错误
错误:
'NSGenericException', reason: Collection <__NSCFSet: 0x16e47100> was mutated while being enumerated
Some times : crash here try managedObjectContext.save()
Sometimes Key value coding Compliant error
我的ViewControllerclass
class ViewController: UIViewController {
var jsonObj:NSDictionary?
var values = [AnyObject]()
@IBOutlet weak var tableView:UITableView!
override func viewDidLoad() {
super.viewDidLoad()
getData()
saveInBD()
NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(self.saved(_:)), name: "kContextSavedNotification", object: nil)
}
//Loding json data from a json file
func getData(){
if let path = NSBundle.mainBundle().pathForResource("countries", ofType: "json") {
do {
let data = try NSData(contentsOfURL: NSURL(fileURLWithPath: path), options: NSDataReadingOptions.DataReadingMappedIfSafe)
do {
jsonObj = try NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.MutableContainers) as? NSDictionary
} catch {
jsonObj = nil;
}
} catch let error as NSError {
print(error.localizedDescription)
}
} else {
print("Invalid filename/path.")
}
}
**Notification reciever**
func saved(not:NSNotification){
dispatch_async(dispatch_get_main_queue()) {
if let data = DatabaseManager.sharedInstance.getAllNews(){
self.values = data
print(data.count)
self.tableView.reloadData()
}
}
}
func saveInBD(){
if jsonObj != nil {
guard let nameArray = jsonObj?["data#"] as? NSArray else{return}
DatabaseManager.sharedInstance.addNewsInBackGround(nameArray)
}
}
//UIButton for re-saving data again
@IBAction func reloadAxn(sender: UIButton) {
saveInBD()
}
}
**Database Manager Class**
public class DatabaseManager{
static let sharedInstance = DatabaseManager()
let managedObjectContext = (UIApplication.sharedApplication().delegate as! AppDelegate).managedObjectContext
private init() {
}
func addNewsInBackGround(arr:NSArray) {
let jsonArray = arr
let moc = managedObjectContext
let privateMOC = NSManagedObjectContext(concurrencyType: .PrivateQueueConcurrencyType)
privateMOC.parentContext = moc
privateMOC.performBlock {
for jsonObject in jsonArray {
let entity = NSEntityDescription.entityForName("Country",
inManagedObjectContext:privateMOC)
let managedObject = NSManagedObject(entity: entity!,
insertIntoManagedObjectContext: privateMOC) as! Country
managedObject.name = jsonObject.objectForKey("name")as? String
}
do {
try privateMOC.save()
self.saveMainContext()
NSNotificationCenter.defaultCenter().postNotificationName("kContextSavedNotification", object: nil)
} catch {
fatalError("Failure to save context: \(error)")
}
}
}
func getAllNews()->([AnyObject]?){
let fetchRequest = NSFetchRequest(entityName: "Country")
fetchRequest.resultType = NSFetchRequestResultType.DictionaryResultType
do {
let results =
try managedObjectContext.executeFetchRequest(fetchRequest)
results as? [NSDictionary]
if results.count > 0
{
return results
}else
{
return nil
}
} catch let error as NSError {
print("Could not fetch \(error), \(error.userInfo)")
return nil
}
}
func saveMainContext () {
if managedObjectContext.hasChanges {
do {
try managedObjectContext.save()
} catch {
let nserror = error as NSError
print("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
您违反了线程限制。
您不能在后台写入 CoreData,而在 MainThread 中读取。
CoreData 的所有操作必须在同一个线程中完成
您可以在后台写入并在主线程中读取(像您一样使用不同的 MOC)。实际上你几乎做对了。
应用程序在 try managedObjectContext.save()
行崩溃,因为 saveMainContext
是从私有 MOC 的 performBlock 中调用的。修复它的最简单方法是将保存操作包装到另一个 performBlock
:
func saveMainContext () {
managedObjectContext.performBlock {
if managedObjectContext.hasChanges {
do {
try managedObjectContext.save()
} catch {
let nserror = error as NSError
print("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
另外两个错误有点棘手。请提供更多信息。什么对象不符合什么键的键值?这很可能是 JSON 解析问题。
第一个错误 ("mutated while being enumerated") 实际上是一个严重的错误。描述非常简单:一个集合在另一个线程上被枚举时被一个线程改变。它发生在哪里?
一个可能的原因(我认为最有可能的原因)是它确实是一个 Core Data 多线程问题。尽管您可以使用多个线程,但您只能在获取它们的线程中使用核心数据对象。如果您将它们传递给另一个线程,您可能 运行 会出现这样的错误。
查看您的代码并尝试找到可能发生这种情况的地方(例如,您是否从其他 类 访问 self.values
?)。不幸的是,我没能在几分钟内找到这样的地方。如果你说这个错误发生在哪个集合枚举上,它会有所帮助。
更新:
P.S。我只是认为错误可能与 saveMainContext
函数有关。它在调用 saved
之前执行。 saveMainContext
在后台线程上执行(在原始代码中,我的意思是),而 saved
在主线程上执行。所以在修复 saveMainContext
之后,错误可能会消失(不过我不是 100% 确定)。
我在使用 private managedObjectContext
在后台保存数据时遇到问题。我是 CoreData 的新手。我正在为 NSManagedObjectContext
使用父子方法,但面临几个问题。
多次点击刷新按钮出现错误
错误:
'NSGenericException', reason: Collection <__NSCFSet: 0x16e47100> was mutated while being enumerated
Some times : crash here
try managedObjectContext.save()
Sometimes Key value coding Compliant error
我的ViewControllerclass
class ViewController: UIViewController {
var jsonObj:NSDictionary?
var values = [AnyObject]()
@IBOutlet weak var tableView:UITableView!
override func viewDidLoad() {
super.viewDidLoad()
getData()
saveInBD()
NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(self.saved(_:)), name: "kContextSavedNotification", object: nil)
}
//Loding json data from a json file
func getData(){
if let path = NSBundle.mainBundle().pathForResource("countries", ofType: "json") {
do {
let data = try NSData(contentsOfURL: NSURL(fileURLWithPath: path), options: NSDataReadingOptions.DataReadingMappedIfSafe)
do {
jsonObj = try NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.MutableContainers) as? NSDictionary
} catch {
jsonObj = nil;
}
} catch let error as NSError {
print(error.localizedDescription)
}
} else {
print("Invalid filename/path.")
}
}
**Notification reciever**
func saved(not:NSNotification){
dispatch_async(dispatch_get_main_queue()) {
if let data = DatabaseManager.sharedInstance.getAllNews(){
self.values = data
print(data.count)
self.tableView.reloadData()
}
}
}
func saveInBD(){
if jsonObj != nil {
guard let nameArray = jsonObj?["data#"] as? NSArray else{return}
DatabaseManager.sharedInstance.addNewsInBackGround(nameArray)
}
}
//UIButton for re-saving data again
@IBAction func reloadAxn(sender: UIButton) {
saveInBD()
}
}
**Database Manager Class**
public class DatabaseManager{
static let sharedInstance = DatabaseManager()
let managedObjectContext = (UIApplication.sharedApplication().delegate as! AppDelegate).managedObjectContext
private init() {
}
func addNewsInBackGround(arr:NSArray) {
let jsonArray = arr
let moc = managedObjectContext
let privateMOC = NSManagedObjectContext(concurrencyType: .PrivateQueueConcurrencyType)
privateMOC.parentContext = moc
privateMOC.performBlock {
for jsonObject in jsonArray {
let entity = NSEntityDescription.entityForName("Country",
inManagedObjectContext:privateMOC)
let managedObject = NSManagedObject(entity: entity!,
insertIntoManagedObjectContext: privateMOC) as! Country
managedObject.name = jsonObject.objectForKey("name")as? String
}
do {
try privateMOC.save()
self.saveMainContext()
NSNotificationCenter.defaultCenter().postNotificationName("kContextSavedNotification", object: nil)
} catch {
fatalError("Failure to save context: \(error)")
}
}
}
func getAllNews()->([AnyObject]?){
let fetchRequest = NSFetchRequest(entityName: "Country")
fetchRequest.resultType = NSFetchRequestResultType.DictionaryResultType
do {
let results =
try managedObjectContext.executeFetchRequest(fetchRequest)
results as? [NSDictionary]
if results.count > 0
{
return results
}else
{
return nil
}
} catch let error as NSError {
print("Could not fetch \(error), \(error.userInfo)")
return nil
}
}
func saveMainContext () {
if managedObjectContext.hasChanges {
do {
try managedObjectContext.save()
} catch {
let nserror = error as NSError
print("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
您违反了线程限制。
您不能在后台写入 CoreData,而在 MainThread 中读取。
CoreData 的所有操作必须在同一个线程中完成
您可以在后台写入并在主线程中读取(像您一样使用不同的 MOC)。实际上你几乎做对了。
应用程序在 try managedObjectContext.save()
行崩溃,因为 saveMainContext
是从私有 MOC 的 performBlock 中调用的。修复它的最简单方法是将保存操作包装到另一个 performBlock
:
func saveMainContext () {
managedObjectContext.performBlock {
if managedObjectContext.hasChanges {
do {
try managedObjectContext.save()
} catch {
let nserror = error as NSError
print("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
另外两个错误有点棘手。请提供更多信息。什么对象不符合什么键的键值?这很可能是 JSON 解析问题。
第一个错误 ("mutated while being enumerated") 实际上是一个严重的错误。描述非常简单:一个集合在另一个线程上被枚举时被一个线程改变。它发生在哪里? 一个可能的原因(我认为最有可能的原因)是它确实是一个 Core Data 多线程问题。尽管您可以使用多个线程,但您只能在获取它们的线程中使用核心数据对象。如果您将它们传递给另一个线程,您可能 运行 会出现这样的错误。
查看您的代码并尝试找到可能发生这种情况的地方(例如,您是否从其他 类 访问 self.values
?)。不幸的是,我没能在几分钟内找到这样的地方。如果你说这个错误发生在哪个集合枚举上,它会有所帮助。
更新:
P.S。我只是认为错误可能与 saveMainContext
函数有关。它在调用 saved
之前执行。 saveMainContext
在后台线程上执行(在原始代码中,我的意思是),而 saved
在主线程上执行。所以在修复 saveMainContext
之后,错误可能会消失(不过我不是 100% 确定)。