iOS对象的归档和解档-Runtime实现

前言

在之前的文章什么是序列化?序列化的作用是什么?iOS中怎么实现序列化?中我们用普通方法实现了对象的序列化操作,也就是归档和解档。

对遵循了NSCoding协议的iOS对象进行归档和解档是我们经常用到的一种数据持久化方式。但是如果对象的属性太多了我们还是对每个属性都实现一遍encodeObject:forKey:decodeObjectForKey:方法,这样写的代码量太多了,会显得很繁琐,今天就教大家用一个简单的方法来实现。

1.创建遵循NSCoding协议的Person对象


#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface Person : NSObject<NSCoding>

@property (copy,   nonatomic) NSString *name;
@property (copy,   nonatomic) NSString *sex;
@property (assign, nonatomic) NSInteger age;
@property (assign, nonatomic) NSInteger height;

@end

NS_ASSUME_NONNULL_END

2.Person.m 代码实现归档和解档

  • runtime归档
- (void)encodeWithCoder:(NSCoder *)coder {
		//告诉系统归档的属性是哪些
	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);//获取到属性的C字符串名称
		NSString *key = [NSString stringWithUTF8String:name];//转成对应的OC名称
		//归档 -- 利用KVC
		[coder encodeObject:[self valueForKey:key] forKey:key];
	}
	free(ivars);
		//在OC中使用了Copy、Creat、New类型的函数,需要释放指针!!(注:ARC管不了C函数)
}
  • runtime解档
- (instancetype)initWithCoder:(NSCoder *)coder {
	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);
			NSString *key = [NSString stringWithUTF8String:name];
				//解档
			id value = [coder decodeObjectForKey:key];
				// 利用KVC赋值
			[self setValue:value forKey:key];
		}
		free(ivars);
	}
	return self;
}


使用runtime的好处不言而喻,无论对象有多少属性都可以通过这个for循环搞定,非常厉害,非常方便。不知道你学会了吗?

猜你喜欢

转载自blog.csdn.net/zjpjay/article/details/86552305