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

    推荐阅读