【问题标题】:Reset all saved data in Core Data model重置 Core Data 模型中所有保存的数据
【发布时间】:2014-12-22 16:01:40
【问题描述】:

我有一种方法允许用户导出我的Core Data model 中所有数据的 .csv 文件。在执行fetchRequest 后,我正在使用美妙的CHCSV Parser 来获取存储的结果。创建 .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),然后返回“管理员”控制屏幕并尝试再次登录,我会得到以下输出:

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 也是如此,但我没有尝试过任何工作。我在AppDelegate 中实现了resetStore 方法来删​​除和重建商店,如下所示:

- (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,它的 in-内部员工)虽然我不想要那个解决方案,但肯定有一种方法可以让我重置我的数据库并继续使用该应用程序,而不必每次都将其关闭。提前致谢。

参考:
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

【问题讨论】:

  • 如何设置self.managedObjectAdmin?或者我应该说,你是如何创建它的?
  • 它是在我的viewDidLoad 方法中创建的,我需要做的就是在导出后再次调用该方法以重新创建managedObject。请看下面我的回答。如此简单的解决方案!

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


【解决方案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 或重新启动应用程序。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-12-27
    • 1970-01-01
    • 2014-11-05
    • 2010-11-07
    • 1970-01-01
    相关资源
    最近更新 更多