Objective-C 中基于RunTime实现的反射
一、反射
反射,一般表现在字符串和Class转换,字符串和内部方法转换,字符串和属性的转换(取值和赋值)。
二、Objective-C中的反射
OC的反射是基于其Runtime实现的。
以执行某个函数为例,我们知道在OC中执行[Stu doSomething]函数,实质上是发送了一个消息给Runtime,然后Runtime再根据这个Class的字符串名和这个函数的字符串名,去匹配真正相应的方法的地址,然后再执行的。所以中间我们可以利用字符串去动态的检测,甚至动态的修改(之前说到的Method Swizzling)。
在OC中,很多Runtime的动态特性的接口大致都已经在NSObject.h中声明,可以自己看下源码。
反射具体表现在:
1.字符串和Class转换,及判断
Class __nullable NSClassFromString(NSString *aClassName); -(BOOL)isKindOfClass:(Class)aClass; -(BOOL)isMemberOfClass:(Class)aClass;
示例:
//通过这样的方式获取class Class name = NSClassFromString(@"Student"); Student *stu = [[name alloc] init];
//判断是否为其子类的对象 Student *stu = [[Student alloc]init]; if([stu isKindOfClass:[Person class]]){ NSLog(@"stu是Person类型或其子类"); }else{ NSLog(@"stu不是Person类型或其子类"); } //判断是否是该class的对象 if([stu isMemberOfClass:[Person class]]){ NSLog(@"stu是Person类型"); }else{ NSLog(@"stu不是Person类型"); }
2.字符串和内部方法转换
- (BOOL)respondsToSelector:(SEL)aSelector 判断类型或对象有没有某个方法
+ (BOOL)instancesRespondToSelector:(SEL)aSelector; //判断静态方法
- (id)performSelector:(SEL)aSelector 动态调用对象的方法 - (BOOL)conformsToProtocol:(Protocol *)aProtocol; 判断对象是否实现某个Protocol协议
示例:
//动态生成方法选择器 SEL sel = NSSelectorFromString(@"setAge:"); //检测是否存在某方法 Student *stu = [[Student alloc]init]; if([stu respondsToSelector:@selector(setAge)]){ NSLog(@"stu 有setAge这个方法"); }else{ NSLog(@"没有"); } //动态动用方法 Student *stu = [[Student alloc]initAge:1]; int age = [stu performSelector:@selector(age)]; NSLog(@"%i",age);//输出1 //动态调用有参数的方法 [stu performSelector:@selector(setAge1:) withObject:@"2"];
3.字符串和属性的转换
OC中属性的反射通过KVC(Key-Value Coding)机制实现,KVC是一种间接访问对象属性的机制,不直接调用getter 和 setter方法,而使用valueForKey 来替代getter 方法,setValue:forKey来代替setter方法。
之前的一篇博客(http://www.cnblogs.com/rayshen/p/5006619.html),在探讨如何把某个对象进行序列化的时候,其实已经使用到KVC,如果某个类遵循NSCoding协议就能编码成NSData(字节流)。
具体KVC使用的示例为:
Persion *persion = [ [Persion alloc] init ]; //不使用KVC persion.name = @"shen" ; //使用KVC的写法 [persion setValue:@"shen" forKey:@"name"];
上面是利用KVC访问类里的某个属性,下面利用KVC直接访问类里的类里的某个属性
//不使用KVC Persion *persion = [ [Persion alloc] init ]; Phone *phone = persion.phone; Battery *battery = phone.battery; //使用KVC Battery *battery = [persion valueForKeyPath: @"phone.battery" ];
对于SetValueForKey,需要小心的是,假如类型匹配错误的情况下,编译会通过,但运行会报错(动态消息机制嘛,能理解)
[persion setValue:[NSNumber numberWithInteger:1] forKey:@"name"]; // 编译并运行,但报错 persion.name = [NSNumber numberWithInteger:1]; // 不能编译
参考博客:
http://blog.csdn.net/victormokai/article/details/19631359
http://www.tuicool.com/articles/2aYfy2
http://www.cnblogs.com/ygm900/archive/2013/01/16/2862676.html
http://www.cnblogs.com/yjmyzz/archive/2011/02/28/1967451.html
http://www.cnblogs.com/yaski/archive/2009/04/05/1429735.html
___________________________________________________
专注iOS/前端开发,广泛涉猎多种平台和技术,欢迎交流
可以在微博关注并@沈z伟