前言
在iOS項目中,用戶信息等不大不小的數據我們一般會歸檔到沙盒文件中,但當需要歸檔的屬性較多時,或后續添加、減少屬性時,都要操作encodeWithCoder:
initWithCoder:
兩個方法。
在你的項目中,歸檔的代碼可能是這樣的:
- (void)encodeWithCoder:(NSCoder *)encoder {
[encoder encodeObject:_uuid forKey:@"uuid"];
[encoder encodeObject:self.token forKey:@"token"];
[encoder encodeObject:_username forKey:@"username"];
[encoder encodeObject:_avatar forKey:@"avatar"];
[encoder encodeObject:_realname forKey:@"realname"];
[encoder encodeObject:_gender forKey:@"gender"];
[encoder encodeObject:_school forKey:@"school"];
[encoder encodeObject:_graduateTime forKey:@"graduateTime"];
[encoder encodeObject:_examType forKey:@"examType"];
[encoder encodeObject:_examArea forKey:@"examArea"];
[encoder encodeObject:_nickname forKey:@"nickname"];
[encoder encodeObject:_studyPhase forKey:@"studyPhase"];
[encoder encodeObject:_subject forKey:@"subject"];
[encoder encodeObject:_consignee forKey:@"consignee"];
[encoder encodeObject:_mobileno forKey:@"mobileno"];
[encoder encodeObject:_province forKey:@"province"];
[encoder encodeObject:_city forKey:@"city"];
[encoder encodeObject:_area forKey:@"area"];
[encoder encodeObject:_street forKey:@"street"];
[encoder encodeObject:_gold forKey:@"gold"];
}
雖然com+c
com+v
很簡單,可是看著還是比較冗余。
runtime幾個函數介紹
Ivar *class_copyIvarList(Class cls, unsigned int *outCount);
作用:拷貝一個類的實例變量(不論是.h還是.m文件)
-
參數:
-
cls
:指定的Class對象 -
outCount
:int型的指針,實例變量的個數,函數內部會對其進行賦值操作
-
返回值:返回一個指針數組,元素是Ivar類型。可以對其進行遍歷
const char *ivar_getName(Ivar v);
- 作用:獲取實例變量的名稱
- 參數:
v
:Ivar類型的實例變量 - 返回值:C的字符串,可以通過
[NSString stringWithUTF8String:cString]
轉為OC字符串
runtime函數演示
先創建一個Person類,并分別在.h文件
類extension
類implementation
三個地方聲明幾個實例變量,如下:
// Person.h
@interface Person : NSObject
@property (nonatomic, copy) NSString *firstName;
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, assign) BOOL gender;
@end
// Person.m
@interface Person ()
@property (nonatomic, assign) CGFloat height;
@end
@implementation Person {
NSString *_lastName;
}
@end
我們在VC
中導入<objc/runtime.h>
,并使用以上介紹的函數書寫以下代碼,獲取Person
類的實例變量數及名稱
- (void)viewDidLoad {
[super viewDidLoad];
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([Person class], &count);
NSLog(@"%d", count);
for (int i=0; i<count; i++) {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
NSLog(@"%@", key);
}
}
先自己考慮下打印結果
打印如下:
2017-09-29 15:16:55.372 Demo[2981:241191] 5
2017-09-29 15:16:55.372 Demo[2981:241191] _lastName
2017-09-29 15:16:55.373 Demo[2981:241191] _gender
2017-09-29 15:16:55.373 Demo[2981:241191] _firstName
2017-09-29 15:16:55.373 Demo[2981:241191] _age
2017-09-29 15:16:55.373 Demo[2981:241191] _height
完全符合預想。
KVC
只是簡單的setValue:forKey:
valueForKey:
操作,不再贅述。
歸檔解檔實現
通過以上介紹,相信大家也猜到怎么實現了。
原理:
歸檔時,通過以上兩個函數獲取到所有實例變量名稱,再通過KVC
獲取到對應的value
,進行歸檔。解檔時,通過獲取到的名稱 使用KVC
進行賦值。
代碼實現:
- (instancetype)initWithCoder:(NSCoder *)decoder {
if (self = [super init]) {
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 = [decoder decodeObjectForKey:key];
[self setValue:value forKey:key];
}
free(ivars);
}
return self;
}
- (void)encodeWithCoder:(NSCoder *)encoder {
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 = [self valueForKey:key];
[encoder encodeObject:value forKey:key];
}
free(ivars);
}
不要忘記釋放堆內存ivars,不然會造成內存泄露。
這樣代碼看起來就不那么冗余了,而且不管以后增加/減少屬性,都不需要改動encodeWithCoder:
initWithCoder:
的代碼了。