Ошибка утверждения во второй раз при добавлении/удалении

У меня странная проблема: у меня есть DetailViewController внутри SplitViewController, который содержит кнопку. Эта кнопка имеет переход Popover к NavigationController и UITableViewController, который действует как блокнот. В первый раз, когда я открываю всплывающее окно, блокнот работает отлично, как и ожидалось. Однако, если я закрою всплывающее окно, а затем снова открою его, произойдет сбой при добавлении или удалении заметки. Функциональность восстанавливается после перезапуска приложения.

Добавить виноватый код:

[_objects insertObject:newNote atIndex:0];

Удалить виноватый код:

[_objects removeObjectAtIndex:indexPath.row];

Почти весь класс NotesViewController.m:

#import "NotesTableViewController.h"

@interface NotesTableViewController ()
{
NSMutableArray *_objects;
BOOL firstRun;
}

@end

@implementation NotesTableViewController
@synthesize openSubject;

- (id)initWithStyle:(UITableViewStyle)style
{
self = [super initWithStyle:style];
if (self) {
    // Custom initialization
}
return self;
}

- (void)awakeFromNib
{
self.clearsSelectionOnViewWillAppear = NO;
self.preferredContentSize = CGSizeMake(320.0, 600.0);
[super awakeFromNib];
}

- (void)viewDidLoad
{
[super viewDidLoad];

// Load the notes data
// Create the key
NSString *partOfKey = @"-notes";
NSString *notesKey = [NSString stringWithFormat:@"%@%@", openSubject, partOfKey];

// Do any additional setup after loading the view, typically from a nib
self.navigationItem.leftBarButtonItem = self.editButtonItem;

// Register a class or nib file using registerNib:forCellReuseIdentifier
// o registerClass:forCellReuiseIdentifier: method before calling this method
[self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"Cell"];

// Load the _objects from NSUserdefaults
_objects = nil;
_objects = [[NSUserDefaults standardUserDefaults] objectForKey:notesKey];

if (openSubject.length == 0) {
    // There's currently no subject open, write it in the navigationbar
    // Prompt = open subject
    // Title = notes header
    self.navigationItem.prompt = @"No subject selected";
    self.navigationItem.title = @"My Notes";
} else {
    // Open the subject

    // Prompt = notes header
    // Title = open subject
    self.navigationItem.prompt = openSubject;
    self.navigationItem.title = @"My notes";
}

// Uncomment the following line to preserve selection between presentations.
// self.clearsSelectionOnViewWillAppear = NO;
}



- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (IBAction)addNote:(id)sender {
// Create a new note
if (openSubject.length == 0) {
    // The openSubject is nil, can't add a subject - tell the user
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle: @"No subject" message: @"Please select a subject prior to adding a note" delegate: nil cancelButtonTitle:@"OK" otherButtonTitles:nil]; [alert show];
} else {
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"New note" message:@"Enter a note" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Add", nil];
    alert.alertViewStyle = UIAlertViewStylePlainTextInput;
    [alert show];
}
}

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
// The user created a new subject, add it
if (buttonIndex == 1) {
    // Get the input text
    NSString *newNote = [[alertView textFieldAtIndex:0] text];

    // Check if the note already exist
    if ([_objects containsObject:newNote]) {
        // Tell the user this note already exists
        UIAlertView *alert = [[UIAlertView alloc] initWithTitle: @"Already exists" message: @"This note already exist, sorry" delegate: nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
        [alert show];
    } else {
        // The note doesn't exist, add it
        // Initialize objects
        if (!_objects) {
            _objects = [[NSMutableArray alloc] init];
        }

        // Add
        [_objects insertObject:newNote atIndex:0];
        NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:0];
        [self.tableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];

        // Save the new _objects
        [self saveObjects];
    }
}
}

-(void)saveObjects {
// Create the key
NSString *partOfKey = @"-notes";

    // Save the new objects
NSString *notesKey = [NSString stringWithFormat:@"%@%@", openSubject, partOfKey];
[[NSUserDefaults standardUserDefaults] setObject:_objects forKey:notesKey];
}

#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
#warning Potentially incomplete method implementation.
// Return the number of sections.
return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
// Return the number of rows in the section.
    return _objects.count;
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewCell *cell = [self.tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath];

if (cell == nil) {
    cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"Cell"];
    cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
}

NSString *object = _objects[indexPath.row];
cell.textLabel.text = [object description];
return cell;
}

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath
{
if (editingStyle == UITableViewCellEditingStyleDelete) {
    [_objects removeObjectAtIndex:indexPath.row];
    [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
}

Вывод отладчика:

2014-05-21 20:43:14.564 myApp[263:60b] viewDidAppear 21-05-2014 20:43:15.093 myApp[263:60b] viewDidAppear 21-05-2014 20:43:29.741 myApp[263:60b] * Завершение работы приложения из-за неперехваченного исключения "NSInternalInconsistencyException", причина: "-[__NSCFArray insertObject:atIndex:]: изменяющий метод отправлен в неизменяемый объект" * Стек вызовов первого броска: (0x2d95bfd3 0x38440ccf 0x2d95bf15 0x2d8cfa93 0xf6e29 0x3038eb29 0x3038e7fb 0x3029605f 0x30348377 0x301f76f5 0x3017055b 0x2d9272a5 0x2d924c49 0x2d924f8b 0x2d88ff0f 0x2d88fcf3 0x32789663 0x301db16d 0xf9411 0x3894dab7) Libc ++ abi.dylib: завершение с неперехваченного исключением типа NSException (lldb)

Пожалуйста, помогите мне решить это, я понятия не имею.

заранее спасибо


person user3653447    schedule 21.05.2014    source источник


Ответы (1)


Когда вы возвращаете свой массив из NSUserDefaults, он возвращает неизменяемый массив. Вам нужно будет получить mutableCopy или выполнить _objects = [NSMutableArray arrayWithArray:[[NSUserDefaults standardUserDefaults] objectForKey:notesKey]];

Вот ссылка на этот ответ

person Christopher Jones    schedule 21.05.2014
comment
Вы, сэр, только что сделали мой день! СПАСИБО - понятия не имею, как сильно это меня беспокоило :) - person user3653447; 21.05.2014