OC运行时内省的4个方法
【OC运行时内省的4个方法】对象在运行时获取其类型的能力称为内省。内省可以有多种方法实现。
OC运行时内省的4个方法:
判断对象类型:
-(BOOL) isKindOfClass: 判断是否是这个类或者这个类的子类的实例
-(BOOL) isMemberOfClass: 判断是否是这个类的实例
判断对象or类是否有这个方法
-(BOOL) respondsToSelector: 判读实例是否有这样方法
+(BOOL) instancesRespondToSelector: 判断类是否有这个方法
在 Objective-C 中,id类型类似于(void*) ,可以指向任何类的对象,但在运行时对象的类型不再是id,而是该对象真正所属的类。
Person *person = [[Person alloc] init];
NSArray *arr = @[person];
idobj = arr[0];
//OC集合中取出的对象都是id类型
此时可通过
BOOLisPersonClass = [objisKindOfClass: [Person class] ];
来判断obj是否Person类型或其子类的对象。
在 Objective-C 中,用父类类型定义的指针,可以指向其子类的对象,但在运行时对象真实类型会是子类。
//例如 Boy是Person的子类,现定义:
Person*p = [[Boy alloc] init];
可通过 BOOLisBoy = [pisMemberOfClass: [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方法。
*/
推荐阅读
- 热闹中的孤独
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- 放屁有这三个特征的,请注意啦!这说明你的身体毒素太多
- 一个人的旅行,三亚
- 布丽吉特,人生绝对的赢家
- 慢慢的美丽
- 尽力
- 一个小故事,我的思考。
- 家乡的那条小河
- 《真与假的困惑》???|《真与假的困惑》??? ——致良知是一种伟大的力量