利用runtime进行归档解档

做过iOS开发的应该都知道数据的本地化的方式,其中归档就是其中的一种。说实话,在本人在项目中并不是很常用归档来进行数据的本地存储。今天之所以写这篇博客是因为最近了解到原来我们还能利用runtime进行归档和解档

先来看一下我们之前的归档、解档代码:

例如我们要对Person这个拥有name和age属性的类的某个对象进行归档,其中Person类中的代码:

#import "Person.h"


@interface Person ()<NSCoding>


@end

@implementation Person


- (void)encodeWithCoder:(NSCoder *)aCoder{

    

    [aCoder encodeObject:_name forKey:@"name"];

    [aCoder encodeObject:_age forKey:@"age"];

}


- (instancetype)initWithCoder:(NSCoder *)coder

{

    self = [super init];

    if (self) {

        _name = [coder decodeObjectForKey:@"name"];

        _age = [coder decodeObjectForKey:@"age"];

    }

    return self;

}

@end

进行归档和解档的代码:

- (IBAction)writeAction:(id)sender {

    NSLog(@"归档");

    Person * person = [Person new];

    person.name = @"梁森";

    person.age = @"26";

    

    NSString * temp = NSTemporaryDirectory();

    NSString * filePath = [temp stringByAppendingPathComponent:@"person.m"];

    NSLog(@"%@", filePath);

    [NSKeyedArchiver archiveRootObject:person toFile:filePath];

}


- (IBAction)readAction:(id)sender {

    NSLog(@"解档");

    NSString * temp = NSTemporaryDirectory();

    NSString * filePath = [temp stringByAppendingPathComponent:@"person.m"];

    

    Person * person = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];

    NSLog(@"%@  %@", person.name, person.age);

    

}

利用runtime进行归档、解档的归档、解档代码和普通的解档、归档代码一样,不一样的地方就是Person类中的代码:

#import "Person.h"

// 引入运行时头文件

#import <objc/runtime.h>

@interface Person()<NSCoding>


@end

@implementation Person


- (void)encodeWithCoder:(NSCoder *)aCoder{

    //属性的个数

    unsigned int count = 0;

    Ivar * ivars = class_copyIvarList([Person class], &count);

    for (int i = 0; i < count; i++) {

        //取出对应的Ivar

        Ivar ivar = ivars[i];

        //拿到名称

        const char * name = ivar_getName(ivar);

        //OC 字符串

        NSString * key = [NSString stringWithUTF8String:name];

        //归档

        [aCoder encodeObject:[self valueForKey:key] forKey:key];

    }

}

- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder{

    self = [super init];

    if (self) {

        //属性的个数

        unsigned int count = 0;

        Ivar * ivars = class_copyIvarList([Person class], &count);

        for (int i = 0; i < count; i++) {

            //取出对应的Ivar

            Ivar ivar = ivars[i];

            //拿到名称

            const char * name = ivar_getName(ivar);

            //OC 字符串

            NSString * key = [NSString stringWithUTF8String:name];

            //解档

            id value = [aDecoder decodeObjectForKey:key];

            //KVC--设置值到属性上面!!!

            [self setValue:value forKey:key];

        }

    }

    return self;

}

@end

看起来好像利用runtime的代码反而更多了,其实不然。如果某个类的属性特别多,假设Person这个类有十个属性,那么我们利用之前的方式,Person中的代码就会很多而且都是一些重复的代码。而我们在使用runtime之后不管我们要进行归档的某个类有多少个属性,这个类中的代码都是一样的多。

解读利用runtime归档Person中的代码:

利用runtime中的某些方法我们获取某个类的所有属性个数以及属性名称,然后利用了for循环对类中的所有属性进行归档、解档的操作。同时还利用到了KVC设置值到属性上面。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值