0

所以我有一个对象,我在其中实现了 NS Coding。我整天都在为此绞尽脑汁。据我了解,因为我已经实现了 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];
4

1 回答 1

1

在您的initWithCoder:方法中,我发现缺少两件事 - 分配[super init]to的结果self,以及保留或复制直接分配给实例变量的任何对象。您可能可以在不将 [super init] 分配给 self 的情况下逃脱,但不保留实例变量将是灾难性的。

在手动引用计数环境中,除非调用转移所有权的方法,否则对象不会获得分配给实例变量的值的所有权。这些方法是newallocretaincopy/ mutableCopy(也是 Core Foundation 创建和复制函数)。使用首字母缩写词 NARC 作为它们的助记符。由于创建分配给 ivars ( NSKeyedUnarchiver) 的对象的对象和对象本身都不拥有它们,因此它们被释放。为避免这种情况,请保留或复制您分配给 ivars 的对象。如果您怀疑分配的对象可能是可变的并且可能被另一个对象突变,则应该进行复制以防止不必要的突变。否则,保留这些值。最后,确保您通过调用放弃这些对象的所有权release在 中dealloc,或在对它们进行任何进一步分配之前(应通过综合或手动实现的 setter 方法完成)。

最后一件事 - 您可以使用NSKeyedArchiverNSKeyedUnarchiver方法archiveRootObject:toFile:unarchiveObjectWithFile:而不是处理文件处理程序和NSFileManager. 这适用于简单的情况,尽管它的错误报告很差。

于 2013-04-02T01:38:40.457 回答