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];
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方法。
*/

你可能感兴趣的:(OC运行时内省的4个方法)