所以我有一个对象,我已经实现了NS编码。这一整天我一直在震撼我的大脑。据我所知,因为我已经实现了NSCoding,所以当在数组本身上调用archive / unarchive方法时,应该对数组的索引进行编码和解码。所以要么我对它的工作原理的理解是关闭的,要么我的实现中有一些错误。
文件被写入,数组被反序列化并显示它有一个对象。但是数组中的类没有数据。
实现如下:
#pragma mark NSCoding
#define kTitle @"Title"
#define kDescription @"Description"
#define kIsCompleted @"IsCompleted"
#define kDate @"Date"
#define kImage @"Image"
-(id)initWithCoder:(NSCoder *)decoder
{
[super init];
title = [decoder decodeObjectForKey:kTitle];
description = [decoder decodeObjectForKey:kDescription];
isCompleted = [decoder decodeBoolForKey:kIsCompleted];
date = [decoder decodeObjectForKey:kDate];
image = [decoder decodeObjectForKey:kImage];
return self;
}
-(void) encodeWithCoder:(NSCoder *) encoder
{
[encoder encodeObject:title forKey:kTitle];
[encoder encodeObject:description forKey:kDescription];
[encoder encodeBool:isCompleted forKey:kIsCompleted];
[encoder encodeObject:date forKey:kDate];
[encoder encodeObject:image forKey:kImage];
}
@end
编码看起来像这样,有一些故障排除更改:
-(void) writeFile
{
NSData *encodedObject;
encodedObject = [NSKeyedArchiver archivedDataWithRootObject: bucketItems];
//NSFileHandle *file;
// file = [NSFileHandle fileHandleForWritingAtPath:filePath];
//if(true)//file == nil)
//{
//NSLog(@"Failed to open a file handle for writing. Attempting to create a file.");
//[[NSFileManager defaultManager] createFileAtPath:filePath contents:nil attributes:nil];
//file = [NSFileHandle fileHandleForWritingAtPath:filePath];
//if(file == nil)
//{
// NSLog(@"Unable to create new file.");
// }
//}
[encodedObject writeToFile:self.filePath atomically:YES];
//[file writeData: encodedObject];
//[file closeFile];
}
然后在这里进行解码,程序启动,检查是否有保存的数据,然后加载
NSFileHandle *file = [NSFileHandle fileHandleForReadingAtPath:filePath];
NSData *fileData;
fileData = [file readDataToEndOfFile];
bucketItems = [NSKeyedUnarchiver unarchiveObjectWithData:fileData];
NSLog(@"Data file found: loading");
[file closeFile];
答案 0 :(得分:1)
在initWithCoder:
方法中,我发现缺少两件事 - 将[super init]
的结果分配给self
,并保留或复制直接分配给实例变量的任何对象。你可以在没有为自己分配[super init]的情况下离开,但不保留实例变量将是灾难性的。
在手动引用计数环境中,除非调用转移所有权的方法,否则对象不会获取分配给实例变量的值的所有权。这些方法是new
,alloc
,retain
和copy
/ mutableCopy
(也是Core Foundation创建和复制功能)。使用首字母缩略词NARC作为那些的助记符设备。由于创建分配给ivars(NSKeyedUnarchiver
)的对象的对象既不拥有它们,也不拥有对象本身,因此它们被释放。为避免这种情况,请保留或复制分配给ivars的对象。如果您怀疑被分配的对象可能是可变的并且可能被另一个对象变异,则应该进行复制以防止不必要的变异。否则,保留值。最后,通过调用release
中的dealloc
,或者在对它们进行任何进一步的赋值之前(应该通过合成或手动实现的setter方法完成),确保放弃对这些对象的所有权。
最后一件事 - 您可以使用NSKeyedArchiver
和NSKeyedUnarchiver
方法archiveRootObject:toFile:
和unarchiveObjectWithFile:
,而不是处理文件处理程序和NSFileManager
。虽然错误报告很差,但这对于简单的案例很有效。