对象在运行时获取其类型的能力称为内省。内省可以有多种方法实现。
OC运行时内省的4个方法:
判断对象类型:
-(BOOL) isKindOfClass: 判断是否是这个类或者这个类的子类的实例
-(BOOL) isMemberOfClass: 判断是否是这个类的实例
判断对象or类是否有这个方法
-(BOOL) respondsToSelector: 判读实例是否有这样方法
+(BOOL) instancesRespondToSelector: 判断类是否有这个方法
在 Objective-C 中,id类型类似于(void*) ,可以指向任何类的对象,但在运行时对象的类型不再是id,而是该对象真正所属的类。
Person *person = [[Person alloc] init];
NSArray *arr = @[person];
id obj = arr[0]; //OC集合中取出的对象都是id类型
此时可通过
BOOL isPersonClass = [obj isKindOfClass: [Person class] ];
来判断obj是否Person类型或其子类的对象。
在 Objective-C 中,用父类类型定义的指针,可以指向其子类的对象,但在运行时对象真实类型会是子类。
//例如 Boy是Person的子类,现定义:
Person *p = [[Boy alloc] init];
可通过 BOOL isBoy = [p isMemberOfClass: [Boy class] ];
判断Person *类型的p是否是Boy类型。
在使用代理调用代理方法时,并不知道代理对象有没有实现对应方法,如未实现就会报方法找不到的错误。此时最好使用respondsToSelector:先判断下,然后调用会比较安全。
另外,可以用来判断方法的版本,避免找不到方法报错。
例如:iOS8以后UIApplication新增注册远程通知的方法,如果ios7及以下系统中调用[application registerForRemoteNotifications];立即会崩溃。
@interface UIApplication (UIRemoteNotifications)
- (void)registerForRemoteNotifications NS_AVAILABLE_IOS(8_0);
@end
此时可以用respondsToSelector来判断:
if ([application respondsToSelector:@selector(registeredForRemoteNotifications)])
{ //IOS8
[application registerForRemoteNotifications];
}
else { // ios7
#pragma clang diagnostic push
#pragma clang diagnostic ignored"-Wdeprecated-declarations"
//夹在3个#pragma中间的不报方法过期警告
[application registerForRemoteNotificationTypes:(UIRemoteNotificationTypeBadge |
UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert)];
#pragma clang diagnostic pop
}
/*
如果不知道某个方法在哪个版本出现的,就可以用respondsToSelector来判断以避免崩溃。
当然也可以先判断[UIDevice currentDevice].systemVersion.floatValue >= 8.0,
再调用registerForRemoteNotifications方法。
*/
网友评论