iOS iOS开发将来跳槽用

一句话 利用runtime轻松实现 归档解档

2017-03-31  本文已影响399人  CocoaJason

先奉上demo的地址

https://github.com/MyHZ/HZCoding.git
使用cocoapods
pod 'NSObject+HZCoding'

我们在开发中存储数据的时候,经常要用的就是归档和解档,我们会在需要操作的对象所在的类中实现两个方法。

-(void)encodeWithCoder:(NSCoder *)aCoder;
-(instancetype)initWithCoder:(NSCoder *)aDecoder;

在这两个方法中,我们会分别调用如下两个方法对每个属性进行操作

- (void)encodeObject:(nullable id)objv forKey:(NSString *)key;
- (nullable id)decodeObjectForKey:(NSString *)key;

当然还有其他类型的

- (void)encodeBool:(BOOL)boolv forKey:(NSString *)key;
- (void)encodeInt:(int)intv forKey:(NSString *)key;

然后将需要存储的属性 一个一个的写进去,而且其中的key又是NSString类型,我们还要保证 归档和解档时 key是一致的。

上面的方法,是我们一贯的做法。

开发中,当有一两个属性的时候(如下),很简单,几句话复制一下,再改属性和key值。

@property(nonatomic,copy) NSString *name;
@property(nonatomic,assign) int age;

但是我们在实际开发中,当然不止有这一两个属性,也不只是NSString和int类型,我们难道要复制 属性个数 * 2 遍吗?

@property(nonatomic,assign) int age;
@property(nonatomic,assign) int age1;
@property(nonatomic,assign) int age2;
@property(nonatomic,assign) int age3;

当然,有人愿意做,那这也是没办法,有耐心,但是出错了,就有的受了。


接下来,我们就用runtime来实现 归档解档

1.我们新建一个Person类,最重要的当然是需要

<NSCoding>协议了

@interface Person : NSObject<NSCoding>

@property(nonatomic,copy) NSString *name;
@property(nonatomic,assign) int age;
@property(nonatomic,assign) int age1;
@property(nonatomic,assign) int age2;
@property(nonatomic,assign) int age3;

@end

重点在于:runtime
首先是 归档
Peroson.m中

-(void)encodeWithCoder:(NSCoder *)aCoder
{
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([Person class], &count);
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        const char *name = ivar_getName(ivar);
        NSString *key = [[NSString alloc]initWithUTF8String:name];
        [aCoder encodeObject:[self valueForKey:key] forKey:key];
    }
}

class_copyIvarList 获的所有的属性
我们通过runtime获取到所有的属性名(name),
然后得到我们需要归档时用到的key(NSString *key),
再使用 KVC 获得归档时用到的encodeObject ([self valueForKey:key] key在刚才已经得到了),

  //归档:
  [aCoder encodeObject:[self valueForKey:key] forKey:key];

解档

-(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 = ivars[i];
            const char *name = ivar_getName(ivar);
            NSString *key = [[NSString alloc]initWithUTF8String:name];
            id value = [aDecoder decodeObjectForKey:key];
            [self setValue:value forKey:key];
        }
        
    }
    return self;
}

核心部分(获取属性名 key)都是和归档是一样的,在接档的时候,同样是使用KVC 获得属性值 [self setValue:value forKey:key];


接下来,我们就可以直接使用Person 类了

归档

 Person *person = [Person new];
    person.name = @"zhangsan";
    person.age = 18;
    
    NSString *temp = NSTemporaryDirectory();
    temp = [temp stringByAppendingPathComponent:@"zhangsan.plist"];
    [NSKeyedArchiver archiveRootObject:person toFile:temp];

解档

NSString *temp = NSTemporaryDirectory();
    temp = [temp stringByAppendingPathComponent:@"zhangsan.plist"];
    Person *person = (Person *)[NSKeyedUnarchiver unarchiveObjectWithFile:temp];
    NSLog(@"%@ %d",person.name,person.age);

到此,就算是完成了Person的归档和解档。


好吧,我又该但是了,我们肯定也不只是Person 这一个类吧,然后每个类再复制一次?那当然不是了。

我们会有更简单的方法

1.首先
我们新建一个NSObject的类别
写两个方法
NSObject+HZCoding

@interface NSObject (HZCoding)

-(void)HZ_encode:(NSCoder *)aCoder;
-(void)HZ_decode:(NSCoder *)aDecoder;

@end

2.接下来 我们在.m中导入

#import <objc/message.h>

3.然后我们实现上面的两个方法

-(void)HZ_encode:(NSCoder *)aCoder
{
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([self class], &count);
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        const char *name = ivar_getName(ivar);
        NSString *key = [[NSString alloc]initWithUTF8String:name];
        [aCoder encodeObject:[self valueForKey:key] forKey:key];
    }
}

-(void)HZ_decode:(NSCoder *)aDecoder
{
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([self class], &count);
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        const char *name = ivar_getName(ivar);
        NSString *key = [[NSString alloc]initWithUTF8String:name];
        id value = [aDecoder decodeObjectForKey:key];
        [self setValue:value forKey:key];
    }
}

@end

4.把我们之前写的Person的那段代码删掉,我们重新搞Person

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

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

毕竟我们在NSObject+HZCoding已经定义了那些个实现的方法,而我们的类 都是继承自NSObject的,那我们直接在自己的类中调用NSObject的方法,不是更方便嘛。
上代码

导入

#import "NSObject+HZCoding.h"
-(void)encodeWithCoder:(NSCoder *)aCoder
{
    [self HZ_encode:aCoder];
}

-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
    if (self = [super init]) {
        [self HZ_decode:aDecoder];
    }return self;
}

好了,这样子又方便了一些,几句话就完成了,我们每个类只需要导入之前的类别(NSObject+HZCoding),调用里面的方法,就可以完成了。


你们猜,有但是没有?

还有

但是 我们还是嫌麻烦,每个类还是要重新写这几行字,我不想写啊。

接下来也不算是黑魔法,也不是什么高大上的万一,我们用宏定义,把这段话给定义好,用到的时候,直接宏定义调用不就好了。
在NSObject+HZCoding.h里面 我们定义下面的内容

#define HZCodingImplementation \
-(void)encodeWithCoder:(NSCoder *)aCoder\
{\
    [self HZ_encode:aCoder];\
}\
-(instancetype)initWithCoder:(NSCoder *)aDecoder\
{\
    if (self = [super init]) {\
        [self HZ_decode:aDecoder];\
    }return self; \
}

接下来就是关键的代码了

重新回到我们的Person类

导入 #import "NSObject+HZCoding.h"
实现:HZCodingImplementation

然后Person.m的代码就变了这样子

#import "Person.h"
#import "NSObject+HZCoding.h"

@implementation Person

HZCodingImplementation

@end

使用的方法,是不是越来越简单了。


总结 总结 总结
总结 总结 总结


用到的时候直接新建一个NSObject的类别,将NSObject+HZCoding里面的内容复制进去就好了。

写的有点乱了,记住用法就好了

协议 <NSCoding>
导入 #import "NSObject+HZCoding.h"
宏定义HZCodingImplementation

OK 就是这么多了

讲的麻烦 ,不如直接看demo了

利用runtime获取对象的所有属性(值),所有方法
http://www.jianshu.com/p/24ff360c0e94

上一篇下一篇

猜你喜欢

热点阅读