Objective-C学习篇第九弹:归档与解档

本文介绍了Objective-C中的归档与解档操作,详细解释了如何使用NSKeyedArchiver和NSKeyedUnarchiver类来保存和恢复对象状态,并提供了多个示例程序,演示了不同场景下的归档过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Foundation框架的归档功能
将对象存储转换为二进制序列的过程成为归档、打包或编码,逆变换称为解档、解码或对象还原。

可以使用NSKeyedArchiver和NSKeyedUnarchiver完成对象的归档和解档操作,而他们都是抽象类NSCoder的子类。

所有可以归档的对象都必须要适用于协议NSCoding。协议NSCoding在Foundation/NSObject.h中定义,NSObject自身并不采用该协议。NSString、NSDictionary等Foundation框架的主要类都适用协议NSCoding。

协议NSCoding按照如下方式声明:

@protocol NSCoding

- (void)encodeWithCoder:(NSCoder *)coder
- (id)initWithCoder:(NSCoder *)coder

归档方法的定义
协议NSCoding中,函数encodeWithCoder:定义了归档自身的方法。

- (void)encodeWithCoder:(NSCoder *)coder
{
    [super encodeWithCoder:coder];
    //超类需要适用NSCoding协议
    [coder encodeObject:对象 forKey:关键词字符串];
    [coder encodeInt:实数变量 forKey:关键词字符串];
}

如果超类不适用协议NSCoding,则不能调用encodeWithCoder:方法。

类自身对包含的实例变量归档。在类没有自己的实例变量且超类中定义了方法encodeWithCoder:的情况下,该方法就不需要在定义了。

通过使用NSString字符串作为键值,可以指定归档解档的内容。让某个类实例归档时,它的实例变量必须指定成不同的键值。在单个对象内部,如果超类使用了一个键值,那么子类中就不能使用该键值。键值只需在同一个类内区分出来即可,不同的类可使用相同的键值。

当对象图有闭环时,同一个对象会重复要求归档,实际上已归档的对象是不用重复归档的。

解档方法的定义

- (instancetype)initWithCoder:(NSCoder *)coder
{
    self = [super initWithCoder:coder];
    //超类不适应于协议NSCoding时,
    //建议使用 self = [super init];
    if (self) {
        self.name = [coder decodeObjectForKey: 键值];
        self.age = [coder decodeIntForKey: 键值];
    }
    return self;
}

归档示例程序:
main.m

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {

    @autoreleasepool {
        Person * p = [[Person alloc] init];
        p.name = @"lu";
        p.age = 18;
        p.weight = 100.0;

        BOOL isSuccess = [NSKeyedArchiver archiveRootObject:p toFile:@"/Users/lu/Desktop/test.plist"];
        if (isSuccess) {
            NSLog(@"归档成功");
        } else {
            NSLog(@"归档失败");
        }
    }
    return 0;

}

解档示例程序:
main.m

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        //解档
        Person * p = [NSKeyedUnarchiver unarchiveObjectWithFile:@"/Users/lu/Desktop/test.plist"];
        NSLog(@"%@",p.name);
        NSLog(@"%d",p.age);
        NSLog(@"%f",p.weight);

    }
    return 0;
}

若类中实例变量为其他类对象,则还需要在其他类中实现encodeWithCoder:和initWithCoder:方法。
示例程序:

Dog.m

#import "Dog.h"

@implementation Dog

- (void)encodeWithCoder:(NSCoder *)coder
{
    [coder encodeObject:self.name forKey:@"name"];
    [coder encodeInt:self.age forKey:@"age"];
}

- (instancetype)initWithCoder:(NSCoder *)coder
{
    self = [super init];
    if (self) {
        self.name = [coder decodeObjectForKey:@"name"];
        self.age = [coder decodeIntForKey:@"age"];
    }
    return self;
}
@end

student.m

#import "Student.h"
#import "Dog.h"

@implementation Student

//归档
- (void)encodeWithCoder:(NSCoder *)coder
{
    [coder encodeObject:self.name forKey:@"name"];
    [coder encodeInt:self.score forKey:@"score"];
    [coder encodeInt:self.number forKey:@"number"];
    [coder encodeObject:self.dog forKey:@"dog"];
}
//解档
- (instancetype)initWithCoder:(NSCoder *)coder
{
    self = [super init];
    if (self) {
        self.name = [coder decodeObjectForKey:@"name"];
        self.score = [coder decodeIntForKey:@"score"];
        self.number = [coder decodeIntForKey:@"number"];
        self.dog = [coder decodeObjectForKey:@"dog"];
    }
    return self;
}

@end

main.m


int main(int argc, const char * argv[]) {

    @autoreleasepool {

        Student * stu = [[Student alloc] init];
        stu.name = @"di";
        stu.score = 70;
        stu.number = 20;

        Dog * d = [[Dog alloc] init];
        d.name = @"Chen";
        d.age = 10;
        stu.dog = d;
        //归档
        BOOL isSuccess = [NSKeyedArchiver archiveRootObject:stu toFile:@"/Users/lu/Desktop/Stu.txt"];
        if (isSuccess) {
            NSLog(@"Yes");
        }
        else {
            NSLog(@"NO");
        }
        //解档
        Student * stu3 = [NSKeyedUnarchiver unarchiveObjectWithFile:@"/Users/lu/Desktop/Stu.txt"];
        NSLog(@"%@", stu3.name);
        NSLog(@"%d", stu3.score);
        NSLog(@"%d", stu3.number); 
        NSLog(@"%@", stu3.dog.name);
        NSLog(@"%d", stu3.dog.age);

    }
    return 0;   
}

对多个对象进行归档,可以先将多个对象存储于NSArray、NSMutableArray、NSDictionary或NSMutableDictionary中,再进行归档。

示例程序:

        NSArray * stuArray = @[stu, stu1, stu2];
        BOOL isSuccess = [NSKeyedArchiver archiveRootObject:stuArray toFile:@"/Users/lu/Desktop/stuArray.txt"];
        if (isSuccess) {
            NSLog(@"Yes");
        }
        else {
            NSLog(@"NO");
        }

最后一种方式利用NSData对多个对象进行归档解档
示例程序

main.m

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        Person * p = [[Person alloc] init];
        p.name = @"dou";
        p.age = 22;
        p.weight = 120.00;

        Person * p2 = [[Person alloc] init];
        p2.name = @"dashu";
        p2.age = 24;
        p2.weight = 140.3;

        NSMutableData * data = [NSMutableData data];

//        根据二进制流创建NSkeyedArchiver对象
        NSKeyedArchiver * archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
//        对对象进行归档操作
        [archiver encodeObject:p forKey:@"person1"];
        [archiver encodeObject:p2 forKey:@"person2"];
//        结束归档
        /*** -[NSKeyedArchiver dealloc]: warning: NSKeyedArchiver deallocated without having had -finishEncoding called on it.
         归档如果没有结束 finishEncoding就不会归档成功,产生文件无法打开    The data couldn't be read because it isn't in the correct format.
         */
        [archiver finishEncoding];

        BOOL isSuccess = [data writeToFile:@"/Users/lu/Desktop/Data.plist" atomically:YES];  
        if (isSuccess) {
            NSLog(@"归档成功!");
        } else {
            NSLog(@"归档失败");
        }

//解档
        NSData * data = [NSData dataWithContentsOfFile:@"/Users/lu/Desktop/Data.plist"];
        NSKeyedUnarchiver * unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
        Person * p = [unarchiver decodeObjectForKey:@"person1"];
        Person * p2 = [unarchiver decodeObjectForKey:@"person2"];
        //结束解档 解档没有finishDecoding无影响
//        [unarchiver finishDecoding];
        NSLog(@"%@", p.name);
        NSLog(@"%d", p.age);
        NSLog(@"%f", p.weight);

        NSLog(@"%@", p2.name);
        NSLog(@"%d", p2.age);
        NSLog(@"%f", p2.weight);
    }
    return 0;
}

总结
这一篇文章我们就说了OC中的归档和解档的相关概念和操作,其实说白了就是将对象写入到文件,和从文件中读取对象。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值