我有一种方法允许用户导出Core Data model
中所有数据的.csv文件。我在执行CHCSV Parser
后使用精彩的fetchRequest
来获取存储的结果。创建.csv后,它将附加到电子邮件并从设备导出。这一切都运行正常,当我删除数据时会出现问题。
我正在使用一种有点流行的技术来删除我的数据(通过流行我的意思是它已被推荐用于我研究过的大量Stack Overflow答案)。我只需获取所有对象并逐个删除它们。
// Create fetchRequest
NGLSAppDelegate *appDelegate = (NGLSAppDelegate *)[[UIApplication sharedApplication]delegate];
NSManagedObjectContext *context = [appDelegate managedObjectContext];
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc]init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"NGLS"
inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSError *error;
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
// Delete objects
for (Model *results in fetchedObjects) {
[context deleteObject:(id)results];
NSLog(@"NGLS objects deleted");
};
我的model
很简单,只有两个entities
,所以我也为其他Admin entity
使用相同的代码。这一切都很好,所有对象都被删除了,我可以通过另一个fetchRequest
来确认这个,它返回每个entity
的对象计数 - 两者的计数都为零。在我执行删除后尝试将数据保存回entity
时,会出现问题。
带有上述代码的ViewController
是"管理员"控制屏幕,用户可以登录并导出数据。因此,当用户登录时,以下是要保存的代码:
// Save login details
[_managedObjectAdmin setValue:user forKey:@"userLogin"];
[_managedObjectAdmin setValue:site forKey:@"siteLocation"];
NSError *error;
[[self.managedObjectAdmin managedObjectContext] save:&error];
然后我执行上面的fetchRequest
并在"导出"按下按钮。完成后,当我尝试登录相同的ViewController
时,输出为:
2014-10-27 12:43:49.653 NGLS[19471:607] <NSManagedObject: 0x7a8c2460> (entity: Admin; id: 0x7a82e3e0 <x-coredata://3C9F3807-E314-439C-8B73-3D4459F85156/Admin/p30> ; data: <fault>)
如果我导航回另一个ViewController
(使用我的navigation controller
),请返回&#34;管理员&#34;控制屏幕,并尝试再次登录,我得到这个输出:
2014-10-27 12:44:04.530 NGLS[19471:607] *** Terminating app due to uncaught exception 'NSObjectInaccessibleException', reason: 'CoreData could not fulfill a fault for '0x7a82e3e0 <x-coredata://3C9F3807-E314-439C-8B73-3D4459F85156/Admin/p30>''
(如果需要,我可以发布完整的日志)。
为尝试解决此问题,我尝试了很多方法。核心数据很复杂,我不太明白我需要做些什么来解决这个问题。我试图删除persistentStore
并再次创建它,与managedObjectContext
一样,但我尝试过的任何内容都没有效果。我在resetStore
中实施了AppDelegate
方法,删除并重建商店,如下所示:
- (void)resetStores {
NSError *error;
NSURL *storeURL = [[_managedObjectContext persistentStoreCoordinator] URLForPersistentStore:[[[_managedObjectContext persistentStoreCoordinator] persistentStores] lastObject]];
// lock the current context
[_managedObjectContext lock];
[_managedObjectContext reset];//to drop pending changes
//delete the store from the current managedObjectContext
if ([[_managedObjectContext persistentStoreCoordinator] removePersistentStore:[[[_managedObjectContext persistentStoreCoordinator] persistentStores] lastObject] error:&error])
{
// remove the file containing the data
[[NSFileManager defaultManager] removeItemAtURL:storeURL error:&error];
//recreate the store like in the appDelegate method
[[_managedObjectContext persistentStoreCoordinator] addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:nil error:&error];//recreates the persistent store
}
[_managedObjectContext unlock];
}
并尝试了这种方法:
- (NSPersistentStoreCoordinator *)resetPersistentStore
{
NSError *error = nil;
if ([_persistentStoreCoordinator persistentStores] == nil)
return [self persistentStoreCoordinator];
_managedObjectContext = nil;
NSPersistentStore *store = [[_persistentStoreCoordinator persistentStores] lastObject];
if (![_persistentStoreCoordinator removePersistentStore:store error:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
// Delete file
if ([[NSFileManager defaultManager] fileExistsAtPath:store.URL.path]) {
if (![[NSFileManager defaultManager] removeItemAtPath:store.URL.path error:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
}
// Delete the reference to non-existing store
_persistentStoreCoordinator = nil;
NSPersistentStoreCoordinator *r = [self persistentStoreCoordinator];
return r;
}
但这些都不奏效。我发现工作是在导出完成后,通过关闭并重新打开应用程序,一切正常。我试图找出关闭/打开关于核心数据的应用程序时发生的过程,但我不知道该寻找什么。我已经研究了Stack Overflow上的许多问题,并尝试了所有解决方案,但对我来说没有任何作用。我真的需要一些帮助,否则我将不得不强迫用户在exit(0);
命令导出后退出应用程序(因为它没有被提交到App Store,它适用于内部员工)虽然我不想要这个解决方案,但我确实有一种方法可以重置我的数据库并继续使用该应用程序,而不必每次都关闭它。提前谢谢。
参考文献:
Deleting all records from Core Data
Reset a Core Data persistent store
How do I delete all objects from my persistent store in Core Data
答案 0 :(得分:1)
我设法通过简单地刷新视图来解决问题。在导出完成后调用我的viewDidLoad
方法,我创建了一个新的managedObject
,可以正常使用,无需离开ViewController
或完全退出应用。
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view from its nib
// Create fetchRequest
NGLSAppDelegate *appDelegate = (NGLSAppDelegate *)[[UIApplication sharedApplication]delegate];
NSManagedObjectContext *context = [appDelegate managedObjectContext];
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Admin" inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSError *error = nil;
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
// Fetch last object
Model *admin = [fetchedObjects lastObject];
// Populate login fields with last stored details
if (admin.userLogin == nil) {
//NSLog(@"Login empty");
} else {
self.usernameField.text = admin.userLogin;
self.siteLocationField.text = admin.siteLocation;
}
// Create new managed object using the Admin entity description
NSManagedObject *ManagedObjectAdmin;
ManagedObjectAdmin = [NSEntityDescription insertNewObjectForEntityForName:@"Admin"
inManagedObjectContext:context];
// Declare managed object
self.managedObjectAdmin = ManagedObjectAdmin;
// Save context
NSError *saveError = nil;
[context save:&saveError];
// ... more stuff
}
解决方案一直都很简单。我会留下这个问题/答案,以便其他人可以从中受益,如果他们发现自己处于我的状况。感谢。
编辑:在导出工作之后的任何地方重新创建managedObjectAdmin
,没有特别需要调用viewDidLoad
方法。导出完成后,我只需创建新的managedObject
并保存context
并继续正常使用该应用,而无需先导航到另一个ViewController
或重新启动应用。