通過 runtime 進(jìn)行歸檔喳篇、解檔很節(jié)省很多工作陪每,我先貼一段常規(guī)的解歸檔的代碼九榔。
/**歸檔:當(dāng)一個對象歸檔(寫入)沙盒時用*/
-(void)encodeWithCoder:(NSCoder *)aCoder
{
[aCoder encodeObject:self.access_token forKey:@"access_token"];
[aCoder encodeObject:self.expires_in forKey:@"expires_in"];
[aCoder encodeObject:self.uid forKey:@"uid"];
[aCoder encodeObject:self.create_Time forKey:@"create_Time"];
[aCoder encodeObject:self.name forKey:@"name"];
[aCoder encodeObject:self.avatar_hd forKey:@"avatar_hd"];
[aCoder encodeObject:self.synopsis forKey:@"synopsis"];
[aCoder encodeObject:self.followers_count forKey:@"followers_count"];
[aCoder encodeObject:self.friends_count forKey:@"friends_count"];
[aCoder encodeObject:self.statuses_count forKey:@"statuses_count"];
}
/**反歸檔:當(dāng)從沙盒讀取一個對象時調(diào)用*/
-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
self = [super init];
if (self) {
self.access_token = [aDecoder decodeObjectForKey:@"access_token"];
self.expires_in = [aDecoder decodeObjectForKey:@"expires_in"];
self.uid = [aDecoder decodeObjectForKey:@"uid"];
self.create_Time = [aDecoder decodeObjectForKey:@"create_Time"];
self.name = [aDecoder decodeObjectForKey:@"name"];
self.avatar_hd = [aDecoder decodeObjectForKey:@"avatar_hd"];
self.synopsis = [aDecoder decodeObjectForKey:@"synopsis"];
self.followers_count = [aDecoder decodeObjectForKey:@"followers_count"];
self.friends_count = [aDecoder decodeObjectForKey:@"friends_count"];
self.statuses_count = [aDecoder decodeObjectForKey:@"statuses_count"];
}
return self;
}
通過這種方式進(jìn)行解歸檔很麻煩,如果需要進(jìn)行歸解檔的屬性變得很多专肪,這種方式就不適用了, 這個時候就需要使用 runtime 來操作堪侯。
這是我自己在學(xué)習(xí)使用 runtime 自動解歸檔的思路及總結(jié):
1. 遵守 NSSecureCoding 協(xié)議
2. 導(dǎo)入 #import <objc/runtime.h>
3. 歸檔操作
- a) 獲取所有實例變量
- b) 遍歷所有實例變量
- c) 獲取實例變量名
- d) 通過
KVC
獲取value
- e) 進(jìn)行編碼
- f) 手動管理內(nèi)存
4. 解檔操作
- a) 獲取所有實例變量
- b) 遍歷所有實例變量
- c) 獲取實例變量名
- d) 進(jìn)行解碼
- d) 通過
KVC
設(shè)置value
- f) 手動管理內(nèi)存
總結(jié):關(guān)鍵的地方在于獲取實例變量名嚎尤,然后再通過 KVC 進(jìn)行操作。
我這里用了 NSSecureCoding
伍宦,沒有用 NSCoding
芽死,是因為 NSSecureCoding
協(xié)議繼承自 NSCoding
,有著比 NSCoding
更加安全的編碼和解碼次洼。
而且关贵,在進(jìn)行解歸檔的時候,原來的方法 archiveRootObject, unarchiveObjectWithFile
在 iOS 11 幾以上是過時的方法卖毁,為了更好的適配揖曾,就使用了NSSecureCoding
,因為是繼承關(guān)系,所以NSCoding
能用的NSSecureCoding
也能用。
需要注意的是炭剪,在使用 NSSecureCoding
的時候要實現(xiàn)supportsSecureCoding
這個方法练链,返回 YES。還有原來的解碼decodeObjectForKey
也要換成另一種姿勢奴拦。用這個- (nullable id)decodeObjectOfClasses:(nullable NSSet<Class> *)classes forKey:(NSString *)key
媒鼓。
代碼實現(xiàn):
//
// Apply.h
// 01-RuntimeSendMessage
//
// Created by Mac on 2019/11/1.
// Copyright ? 2019 Mac. All rights reserved.
//
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface Apply : NSObject<NSSecureCoding>
@property(nonatomic, copy) NSString *name;
@property(nonatomic, strong) NSNumber *age;
@property(nonatomic, copy) NSString *nick;
@end
NS_ASSUME_NONNULL_END
//
// Apply.m
// 01-RuntimeSendMessage
//
// Created by Mac on 2019/11/1.
// Copyright ? 2019 Mac. All rights reserved.
//
#import "Apply.h"
#import <objc/runtime.h>
@implementation Apply
// 歸檔的時候,系統(tǒng)會使用編碼器把當(dāng)前對象編碼成二進(jìn)制流
- (void)encodeWithCoder:(NSCoder *)coder {
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 stringWithUTF8String:name];
// KVC
id value = [self valueForKey:key];
// 編碼
[coder encodeObject:value forKey:key];
}
// 因為是 C 語言的東西错妖,不會自動釋放绿鸣,所以這里需要手動釋放。
free(ivars);
}
// 解檔的時候站玄,系統(tǒng)會把二進(jìn)制流解碼成對象
- (instancetype)initWithCoder:(NSCoder *)coder {
self = [super init];
if (self) {
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 stringWithUTF8String:name];
id value = [coder decodeObjectOfClasses:[NSSet setWithObject:[self class]] forKey:key];
// KVC
[self setValue:value forKey:key];
}
free(ivars);
}
return self;
}
+ (BOOL)supportsSecureCoding {
return YES;
}
@end
在使用的時候:
// 4.自動解歸檔
Apply *apply = [Apply new];
apply.name = @"張三";
apply.age = @18;
apply.nick = @"zhangsan";
Apply *apply_2;
NSString *fileName = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject stringByAppendingPathComponent:@"archive.plist"];
if (@available(iOS 11.0, *)) {
NSData *data_1 = [NSKeyedArchiver archivedDataWithRootObject:apply requiringSecureCoding:YES error:nil];
[data_1 writeToFile:fileName atomically:YES];
NSData *data_2 = [[NSData alloc] initWithContentsOfFile:fileName];
apply_2 = [NSKeyedUnarchiver unarchivedObjectOfClass:[Apply class] fromData:data_2 error:nil];
} else {
[NSKeyedArchiver archiveRootObject:apply toFile:fileName];
apply_2 = [NSKeyedUnarchiver unarchiveObjectWithFile:fileName];
}
NSLog(@"name: %@, age: %@, nick: %@", apply_2.name, apply_2.age, apply_2.nick);
看一下打印效果:
通過打印已經(jīng)知道自動解歸檔已經(jīng)實現(xiàn)了枚驻!