所以我有一个在其中实现了 NS 编码的对象。我一整天都在为此绞尽脑汁。据我了解,因为我已经实现了 NSCoding,所以当对数组本身调用存档/取消存档方法时,应该对数组的索引进行编码和解码。因此,要么是我对其工作原理的理解不正确,要么是我的实现存在一些错误。
文件被写入,数组被反序列化并显示其中有一个对象。但数组中的类没有数据。
实现如下所示:
#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];
最佳答案
在您的 initWithCoder:
方法中,我发现缺少两件事 - 将 [super init]
的结果分配给 self
,并保留或复制直接分配给实例变量的任何对象。您也许可以在不将 [super init] 分配给 self 的情况下逃脱,但不保留实例变量将是灾难性的。
在手动引用计数环境中,对象不会获取分配给实例变量的值的所有权,除非调用转移所有权的方法。这些方法是 new
、alloc
、retain
和 copy
/mutableCopy
(也核心基础创建和复制功能)。使用缩写 NARC 作为这些内容的助记符。由于创建分配给 ivars 的对象 (NSKeyedUnarchiver
) 的对象和对象本身都不拥有它们,因此它们被释放。为了避免这种情况,请保留或复制分配给 ivars 的对象。如果您怀疑所分配的对象可能是可变的并且可能被另一个对象突变,则应该进行复制以防止不必要的突变。否则,保留这些值。最后,确保您通过在 dealloc
中调用 release
放弃这些对象的所有权,或者在对它们进行任何进一步分配之前(这应该通过合成或手动实现的 setter 来完成)方法)。
最后一件事 - 您可以使用 NSKeyedArchiver
和 NSKeyedUnarchiver
方法 archiveRootObject:toFile:
和 unarchiveObjectWithFile:
代替处理文件处理程序和 NSFileManager 。尽管错误报告很差,但这对于简单的情况来说效果很好。
关于iOS 序列化/反序列化无法正常工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15754000/