iOS序列化/反序列化无法正常工作

时间:2013-04-02 00:12:55

标签: ios objective-c serialization deserialization

所以我有一个对象,我已经实现了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];

1 个答案:

答案 0 :(得分:1)

initWithCoder:方法中,我发现缺少两件事 - 将[super init]的结果分配给self,并保留或复制直接分配给实例变量的任何对象。你可以在没有为自己分配[super init]的情况下离开,但不保留实例变量将是灾难性的。

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

最后一件事 - 您可以使用NSKeyedArchiverNSKeyedUnarchiver方法archiveRootObject:toFile:unarchiveObjectWithFile:,而不是处理文件处理程序和NSFileManager。虽然错误报告很差,但这对于简单的案例很有效。