Не могу понять, почему происходит сбой моего приложения при использовании NSKeyedArchivers / NSKeyedUnarchivers - PullRequest
2 голосов
/ 09 мая 2011

Я занимаюсь разработкой своего первого приложения iphone «Дневник», в котором используются пользовательские объекты «Entry», содержащие заголовок NSString, текст NSString и NSDate creationDate.Когда я пытаюсь заархивировать NSMutableArray объектов Entry, а затем получить их при следующей загрузке представления, приложение вылетает.Я просмотрел кучу примеров кодов и примеров, использующих NSKeyedArchivers, но все еще не мог понять, почему это происходит.Я предполагаю, что есть проблема с инициализацией массива, который содержит записи, но не уверен ...

Вот код, может быть, вы могли бы найти то, что я постоянно наблюдал ... "

//--------- Entry.m---------------

- (id) initWithCoder:(NSCoder *)aDecoder{

if ((self = [super init])) {
    self.title = [[aDecoder decodeObjectForKey:@"title"] retain];
    self.text = [[aDecoder decodeObjectForKey:@"text"] retain];
    self.created = [[aDecoder decodeObjectForKey:@"created"] retain];
}
return self;

}

- (void) encodeWithCoder:(NSCoder *)aCoder{

[aCoder encodeObject:self.title forKey:@"title"];
[aCoder encodeObject:self.text forKey:@"text"];
[aCoder encodeObject:self.created forKey:@"created"];

}

//-------------- Diary View Controller.m

- (NSString *)dataFilePath {
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, 
     NSUserDomainMask, YES);
NSString *documentsDirectory = [paths objectAtIndex:0];
return [documentsDirectory stringByAppendingPathComponent:kFilename];
}

- (void) writeDataToArchive {
NSMutableData *data = [[NSMutableData alloc] init];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc]
                             initForWritingWithMutableData:data];

[archiver encodeObject:self.entriesArray forKey:@"entriesArray"];
[archiver finishEncoding];
BOOL result = [data writeToFile:[self dataFilePath]  atomically:YES];
[archiver release];
[data release];    
}

- (void)addItem:sender {
int count = [entriesArray count] +1;    
NSString *newEntryTitle = [NSString stringWithFormat:@"Entry %d", count];    
Entry *anEntry = [[Entry alloc] initWithTitle:newEntryTitle text:@"-" 
         created:[NSDate date]];
[entriesArray addObject:anEntry];
[self.tableView reloadData];

[anEntry release];
[self writeDataToArchive];
}

- (void)viewDidLoad
{
[super viewDidLoad];
NSString *filePath = [self dataFilePath];
if ([[NSFileManager defaultManager] fileExistsAtPath:filePath]) {     
     NSData *data = [[NSMutableData alloc]
                     initWithContentsOfFile:[self dataFilePath]];

     NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc]
          initForReadingWithData:data];
     NSMutableArray *array = [unarchiver decodeObjectForKey:@"entriesArray"];     
     entriesArray = [array mutableCopy];
     [array release];
     [unarchiver finishDecoding];
     [unarchiver release];
     [data release];
}
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:
    (NSIndexPath *)indexPath
{
    // ... some other stuff
     NSUInteger row = indexPath.row;
     Entry *entry = [entriesArray objectAtIndex:row];

     cell.textLabel.text = entry.title;
     return cell;
}

Большое спасибо.

Ответы [ 2 ]

3 голосов
/ 09 мая 2011

Когда вы читаете массив обратно с помощью NSKeyedUnarchivers, вы всегда получаете обратно неизменяемую копию.Вам нужно объявить * массив как NSArray или просто избавиться от массива все вместе.

entriesArray = [[unarchiver decodeObjectForKey:@"entriesArray"] mutableCopy]; 

И @JeremyP указывает на другую проблему.Поскольку вы не выделяли или не сохраняли * массив, вы не должны его освобождать.

2 голосов
/ 09 мая 2011

Вы не должны выпускать array в viewDidLoad, потому что вы не являетесь его владельцем.

Ознакомьтесь с Правилами управления памятью какао , поскольку в вашем коде есть несколько других проблем с управлением памятью.В частности,

self.title = [[aDecoder decodeObjectForKey:@"title"] retain];
self.text = [[aDecoder decodeObjectForKey:@"text"] retain];
self.created = [[aDecoder decodeObjectForKey:@"created"] retain];

в вашем методе initWithCoder: все утечки при условии, что свойства сохраняются или копируются.

...