gpt4 book ai didi

ios - 核心数据以一对多关系保存在后台

转载 作者:塔克拉玛干 更新时间:2023-11-02 21:37:19 24 4
gpt4 key购买 nike

在具有一对多关系(一个“测试”,多个“测量”)的核心数据应用程序中,我曾经有这样的代码:

在 AppDelegate.m 中:

- (NSManagedObjectContext *)managedObjectContext
{
if (_managedObjectContext != nil)
return _managedObjectContext;

NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];

if (coordinator != nil)
{
_managedObjectContext = [[NSManagedObjectContext alloc] init];
[_managedObjectContext setPersistentStoreCoordinator:coordinator];
}
return _managedObjectContext;
}

在 TableViewController.m 中:

- (NSManagedObjectContext *)managedObjectContext
{
NSManagedObjectContext *context = nil;
id contextDelegate = [[UIApplication sharedApplication] delegate];

if ([contextDelegate performSelector:@selector(managedObjectContext)])
context = [contextDelegate managedObjectContext];

return context;
}

- (void)saveEntryButton:(id)sender
{
NSManagedObjectContext *context = [self managedObjectContext];

if (self.test)
{
// Update existing test
self.test.number = self.numberTextField.text;
}
else // Create new test
{
self.test = [NSEntityDescription insertNewObjectForEntityForName:@"Test" inManagedObjectContext:context];
self.test.number = self.numberTextField.text;
}

if (isSaving)
{
NSManagedObjectContext *context = [test managedObjectContext];
self.measure = [NSEntityDescription insertNewObjectForEntityForName:@"Measure" inManagedObjectContext:context];
[test addWithMeasureObject:measure];
NSData *newDataArray = [NSKeyedArchiver archivedDataWithRootObject:plotDataArray];
self.measure.dataArray = newDataArray;
}

NSError *error = nil;
// Save the object to persistent store
if (![context save:&error])
{
NSLog(@"Can't Save! %@ %@", error, [error localizedDescription]);
}
}

效果很好,但当然,[NSKeyedArchiver archivedDataWithRootObject:plotDataArray]; 可能需要几秒钟并阻塞 UI,所以我想在后台进行。

我花了几个小时阅读了有关核心数据中并发性的所有内容(我对此还很陌生),但我没有找到任何有关我的问题的信息:如何处理一对多关系背景保存?

到目前为止我尝试了什么:

在 AppDelegate.m 中

- (NSManagedObjectContext *)managedObjectContext
{
if (_managedObjectContext != nil)
return _managedObjectContext;

NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];

if (coordinator != nil)
{
_managedObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
[_managedObjectContext setPersistentStoreCoordinator:_persistentStoreCoordinator];

//_managedObjectContext = [[NSManagedObjectContext alloc] init];
//[_managedObjectContext setPersistentStoreCoordinator:coordinator];
}
return _managedObjectContext;
}

在 TableViewController.m 中

- (void)saveEntryButton:(id)sender
{
NSManagedObjectContext *context = [self managedObjectContext];

if (self.test)
{
// Update existing test
self.test.number = self.numberTextField.text;
}
else // Create new test
{
self.test = [NSEntityDescription insertNewObjectForEntityForName:@"Test" inManagedObjectContext:context];
self.test.number = self.numberTextField.text;

NSError *error = nil;
// Save the object to persistent store
if (![context save:&error])
{
NSLog(@"Can't Save! %@ %@", error, [error localizedDescription]);
}
}

if (isSaving)
{
NSManagedObjectContext *context = [test managedObjectContext];

NSManagedObjectContext *temporaryContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
temporaryContext.parentContext = context;

[temporaryContext performBlock:^{
self.measure = [NSEntityDescription insertNewObjectForEntityForName:@"Measure" inManagedObjectContext:temporaryContext];

[test addWithMeasureObject:measure];
NSData *newDataArray = [NSKeyedArchiver archivedDataWithRootObject:plotDataArray];
self.measure.dataArray = newDataArray;

// push to parent
NSError *error;
if (![temporaryContext save:&error])
{
// handle error
NSLog(@"error");
}


// save parent to disk asynchronously
[context performBlock:^{
[UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
NSError *error;
if (![context save:&error])
{
// handle error
NSLog(@"error");
}
}];

}];

}

}

当然,我收到 SIGABRT 错误,因为“测试”和“测量”不在同一上下文中……我尝试了很多不同的东西,但我真的迷路了。在此先感谢您的帮助。

最佳答案

我在这里看到两个问题:后台异步保存以及如何处理不同上下文中的对象。

首先是关于储蓄。您确定它正在保存自己阻止您的 UI 线程而不是调用 archivedDataWithRootObject 吗?如果保存本身相对较快,您可以考虑在后台队列中仅调用 archivedDataWithRootObject,然后将结果传回主队列,您将在主队列中对 UI 上下文进行保存。

如果还是保存时间过长,可以使用苹果推荐的后台异步保存的方式。你需要两个上下文。一个上下文——我们称之为背景——是私有(private)队列并发类型。它还配置了持久存储协调器。另一个上下文——我们称它为 UI——是主队列并发类型。它配置有背景上下文作为父级。

使用用户界面时,您使用的是 UI 上下文。所以所有的托管对象都是在这个上下文中插入、修改和删除的。然后当你需要保存时:

NSError *error;
BOOL saved = [UIContext save:&error];
if (!saved) {
NSLog(@“Error saving UI context: %@“, error);
} else {
NSManagedObjectContext *parent = UIContext.parentContext;
[parent performBlock:^{
NSError *parentError;
BOOL parentSaved = [parent save:&parentError];
if (!parentSaved) {
NSLog(@“Error saving parent: %@“, parentError);
}
}];
}

UI 上下文的保存非常快,因为它不会将数据写入磁盘。它只是将更改推送到其父级。而且因为父级是私有(private)队列并发类型,并且您在 performBlock 的 block 中进行保存,所以该保存在后台发生,不会阻塞主线程。

现在关于您示例中不同上下文中的不同托管对象。正如您所发现的,您无法将一个上下文中的对象设置为另一个上下文中对象的属性。您需要选择需要进行更改的上下文。然后将其中一个对象的 NSManagedObjectID 传输到目标上下文。然后使用上下文的方法之一从 ID 创建一个托管对象。最后将此对象设置为另一个对象的属性。

关于ios - 核心数据以一对多关系保存在后台,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21204939/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com