TableView 由于核心数据错误而崩溃/冻结

Posted

技术标签:

【中文标题】TableView 由于核心数据错误而崩溃/冻结【英文标题】:TableView Crashing/Freezing Because Of Core Data Error 【发布时间】:2011-03-28 03:28:48 【问题描述】:

当用户单击导航栏中的 + 按钮时,会出现带有文本提示的 UIAlert。然后用户在提示中输入一个字符串,它应该会生成一个新的 UITableViewCell,名称为字符串。

由于某种原因,当我进入此 viewController 的屏幕时,应用程序崩溃了。

它认为它与 ViewDidLoad 中的以下行有关:NSEntityDescription *entity = [NSEntityDescription entityForName:@"Routine" inManagedObjectContext:managedObjectContext];

控制台显示以下内容:Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: '+entityForName: could not locate an NSManagedObjectModel for entity name 'Routine''

我认为我需要使用“名称”而不是常规,但这也不起作用。

这是我的核心数据模型:

这是我的代码:

#import "RoutineTableViewController.h"
#import "AlertPrompt.h"
#import "Routine.h"

@implementation RoutineTableViewController

@synthesize tableView;
@synthesize eventsArray;
@synthesize managedObjectContext;

    - (void)dealloc
    
        [managedObjectContext release];
        [eventsArray release];
        [super dealloc];
    

    - (void)didReceiveMemoryWarning
    
        [super didReceiveMemoryWarning];
    

    #pragma mark - View lifecycle

    - (void)viewDidLoad
    
        UIBarButtonItem * addButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemAdd target:self action:@selector(showPrompt)];
        [self.navigationItem setLeftBarButtonItem:addButton];
        [addButton release];

        UIBarButtonItem *editButton = [[UIBarButtonItem alloc]initWithTitle:@"Edit" style:UIBarButtonItemStyleBordered target:self action:@selector(toggleEdit)];
        self.navigationItem.rightBarButtonItem = editButton;
        [editButton release];

        NSFetchRequest *request = [[NSFetchRequest alloc] init];
        NSEntityDescription *entity = [NSEntityDescription entityForName:@"Routine" inManagedObjectContext:managedObjectContext];
        [request setEntity:entity];

        NSError *error = nil;
        NSMutableArray *mutableFetchResults = [[managedObjectContext executeFetchRequest:request error:&error] mutableCopy];
        if (mutableFetchResults == nil) 
            // Handle the error.
        

        [self setEventsArray:mutableFetchResults];
        [mutableFetchResults release];
        [request release];

        [super viewDidLoad];
    

    -(void)toggleEdit
    
        [self.tableView setEditing: !self.tableView.editing animated:YES];

        if (self.tableView.editing)
            [self.navigationItem.rightBarButtonItem setTitle:@"Done"];
        else
            [self.navigationItem.rightBarButtonItem setTitle:@"Edit"];
    

    -(void)showPrompt
    
        AlertPrompt *prompt = [AlertPrompt alloc];
        prompt = [prompt initWithTitle:@"Add Workout Day" message:@"\n \n Please enter title for workout day" delegate:self cancelButtonTitle:@"Cancel" okButtonTitle:@"Add"];

        [prompt show];
        [prompt release];
    

    - (void)alertView:(UIAlertView *)alertView willDismissWithButtonIndex:(NSInteger)buttonIndex
    
        if (buttonIndex != [alertView cancelButtonIndex])
        
            NSString *entered = [(AlertPrompt *)alertView enteredText];
            if(eventsArray && entered)
            
                [eventsArray addObject:entered];
                [tableView reloadData];
            
        
    

    -(void)addEvent
    
        Routine *routine = (Routine *)[NSEntityDescription insertNewObjectForEntityForName:@"Routine" inManagedObjectContext:managedObjectContext];

        NSError *error = nil;
        if (![managedObjectContext save:&error]) 
            // Handle the error.
        

        [eventsArray insertObject:routine atIndex:0];
        NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:0];
        [self.tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:indexPath]
                              withRowAnimation:UITableViewRowAnimationFade];
        [self.tableView scrollToRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] atScrollPosition:UITableViewScrollPositionTop animated:YES];
    

    - (void)viewDidUnload
    
        self.eventsArray = nil;
        [super viewDidUnload];
    

#pragma mark - Table view data source

    - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
    
        return 1;
    

    - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
    
        return [eventsArray count];
    

    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    
        static NSString *CellIdentifier = @"Cell";

        UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
        if (cell == nil) 
            //cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];
            cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellEditingStyleDelete reuseIdentifier:CellIdentifier] autorelease];
        
        cell.textLabel.text = [self.eventsArray objectAtIndex:indexPath.row];

        return cell;
    
    // Override to support conditional editing of the table view.
    - (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath
    
        // Return NO if you do not want the specified item to be editable.
        return YES;
    

    -(void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath
     

         if (editingStyle == UITableViewCellEditingStyleDelete) 

             // Delete the managed object at the given index path.
             NSManagedObject *eventToDelete = [eventsArray objectAtIndex:indexPath.row];
             [managedObjectContext deleteObject:eventToDelete];

             // Update the array and table view.
             [eventsArray removeObjectAtIndex:indexPath.row];
             [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:YES];

             // Commit the change.
             NSError *error = nil;
             if (![managedObjectContext save:&error]) 
                 // Handle the error.
             
         
         
    @end

【问题讨论】:

你是如何初始化managedObjectContext的?你确定不是 nil 吗? 【参考方案1】:

我没有看到您的 ManagedObjectContext 是在哪里声明并与您现有的数据模型挂钩的。例如,您在哪里声明您的“getter”以便从 persistentStoreCoordinator 访问它。尝试检查您的连接并在 viewDidLoad 上插入。并在此处查看文档步骤:

http://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/CoreData/Articles/cdUsingMOM.html#//apple_ref/doc/uid/TP40005190-SW1

这是一个连接到您的 persistantStoreCoordinator 的示例

- (NSManagedObjectContext *) managedObjectContext 

if (managedObjectContext != nil) 
    return managedObjectContext;


NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
if (coordinator != nil) 
    managedObjectContext = [[NSManagedObjectContext alloc] init];
    [managedObjectContext setPersistentStoreCoordinator: coordinator];

return managedObjectContext;

或者从您正在使用的教程中,您会在应用程序委托的 didFinishLaunching 方法中看到它:

NSManagedObjectContext *context = [self managedObjectContext];
if (!context) 
    // Handle the error.

// Pass the managed object context to the view controller.
rootViewController.managedObjectContext = context;

编辑

在审查您的代码后,您需要做两件事:

1) 编辑您的 AppDelegate 以加载“Curl”模型,而不是“Temp”模型。这是您的 xdatamodel 的名称。

2) 您需要引用您的应用委托的上下文,而不是在本地创建一个。即

CurlAppDelegate *curlAppDelegate = [[UIApplication sharedApplication] delegate];
    NSManagedObjectContext *context = [curlAppDelegate managedObjectContext];

    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Routine" inManagedObjectContext:context];
    [request setEntity:entity];

    NSError *error = nil;
    NSMutableArray *mutableFetchResults = [[context executeFetchRequest:request error:&error] mutableCopy];
    if (mutableFetchResults == nil) 
        // Handle the error.
    

【讨论】:

我正在关注此处发布的教程 (developer.apple.com/library/ios/#documentation/DataManagement/…),他们似乎没有使用您拥有的那些方法。还是我错过了? 我添加了您发布的代码,似乎一切正常,但仍然无法正常工作。 在本教程(非常棒)中,您应该在应用程序委托中设置您的上下文。是你做的吗?并确保 xdatamodel 已保存?你有一个压缩到某个地方的项目链接,我可以调试它吗?这是解释设置的页面:developer.apple.com/library/ios/#documentation/DataManagement/… 谢谢,多米尼克。我要回去复习那个教程,看看我是否遗漏了什么。这是我压缩后的项目的链接:box.net/shared/static/colensfm58.zip。谢谢 感谢 Dominic,修复了崩溃并且视图现在可以正常工作。但由于某种原因,它没有保存数据并在应用程序的下一次启动时获取它。我将再次查看代码,我可能忘记实现获取请求。【参考方案2】:

“NSInternalInconsistencyException”错误与将基础数据模型更改到 Core Data 无法执行 automatic lightweight data migration 的程度有关:您用于存储数据的实际 SQLite 或 plist 文件现在与数据模型的新结构。

要清除它,您可以按照通常的方式从模拟器(或设备,如果您正在测试的地方)中删除应用程序 - 按住应用程序图标直到它摆动,然后点击/单击 X -- 或者从您 Mac 上应用的工作目录中删除文件本身。

~/Library/Application Support/iPhone Simulator/YOUR-IOS-BASE-SDK-HERE/Applications/YOUR-36-BYTE-APP-ID-HERE/Documents

(或者如果不是 Documents,则以您使用的文件夹为准。)

之后,您可以运行该特定错误将消失的应用程序,因为 Core Data 将能够再次从头开始创建文件。

【讨论】:

以上是关于TableView 由于核心数据错误而崩溃/冻结的主要内容,如果未能解决你的问题,请参考以下文章

swift中的核心数据值访问错误

NSFetchedResultsController 崩溃的 TableView 行删除?

重新加载数据时 Tableview 崩溃

-[NSView setFrame:] 由于无效约束而崩溃

核心数据保存竞争条件错误

由于核心数据故障,应用程序在启动时崩溃