重置Core Data模型中的所有已保存数据

时间:2014-10-27 13:01:27

标签: ios objective-c core-data entity nsfetchrequest

我有一种方法允许用户导出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

1 个答案:

答案 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或重新启动应用。