iOS runtime自动归档问题

一 异常信息

[__NSPlaceholderDictionary initWithObjects:forKeys:count:]: attempt to insert nil object from objects[0]'


二 问题原因

2.1 代码块

/**
 *  把对象归档到文件中
 */
- (void)encodeWithCoder:(NSCoder *)encoder
{
	unsigned int count = 0;
	Ivar *ivars = class_copyIvarList([GMAccessToken class], &count);
	
	for (int i = 0; i<count; i++) {
		// 取出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);
}

/**
 *  从文件中解析对象
 */
- (id)initWithCoder:(NSCoder *)decoder
{
	
	if (self = [super init]) {
		unsigned int count = 0;
		Ivar *ivars = class_copyIvarList([GMAccessToken class], &count);
		
		for (int i = 0; i<count; i++) {
			// 取出i位置对应的成员变量
			Ivar ivar = ivars[i];
			
			// 查看成员变量
			// 解析name 时,如果解析出来的有下划线,则需要去掉下划线
			const char *name = ivar_getName(ivar);
			
			// 归档
			NSString *key = [NSString stringWithUTF8String:name];
//			NSString *key = [[NSString stringWithUTF8String:name] substringFromIndex:1];
			id value = [decoder decodeObjectForKey:key];
			GMLog(@"%@====%@",key,value);
			
			// 设置到成员变量身上
			[self setValue:value forKey:key];
		}
		
		free(ivars);
	}
	return self;
}



2.2 输出

GMLog(@"%@====%@",key,value); 
2016-03-30 22:45:40.058 RT[26053:2065800] _access_token====(null)
2016-03-30 22:45:40.059 RT[26053:2065800] _expires_in====(null)
2016-03-30 22:45:40.059 RT[26053:2065800] _expires_time====(null)
2016-03-30 22:45:40.059 RT[26053:2065800] _remind_in====(null)
2016-03-30 22:45:40.059 RT[26053:2065800] _uid====(null)




通过以下代码, 获取key 的时 候是带下划线的,在代码运行时,会把_key 当成属性而不是字符串 key,导致该问题
NSString *key = [NSString stringWithUTF8String:name];


三 解决方法

去掉下划线

NSString *key = [[NSString stringWithUTF8String:name] substringFromIndex:1];





你可能感兴趣的:(iOS runtime自动归档问题)