Objective-C 【动态类型检测&响应方法】

时间:2022-04-16 15:20:17
———————————————————————————————————————————
动态类型检测

代码:

#import <Foundation/Foundation.h>

@interface Animal : NSObject
-(void)run;
-(void)abc;
@end

@implementation Animal
-(void)run
{
    NSLog(@"Animal run!");
}
-(void)abc
{
    NSLog(@"abc!");
}
@end

@interface Dog : Animal
-(void)run;
-(void)eat;
-(void)bark;
@end

@implementation Dog
-(void)run
{
    NSLog(@"Dog run!");
}
-(void)eat
{
    NSLog(@"Dog eat!");
}
-(void)bark
{
    NSLog(@"Dog bark!");
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
        //动态类型检测:
        //*******************************************************
        
//        1)判断某个对象是否是该类的实例对象,或者是其子类的实例对象(对象和类)
//        isKindOfClass使用格式:   [对象 isKindOfClass:类对象];
       
        Animal *ani=[Animal new];
        
        //BOOL isIstance=[ani isKindOfClass:[ani class]];
        BOOL isInstance=[ani isKindOfClass:[Animal class]];
        //上面两种写法都行,因为可以用 类类型的实例对象/类名 去调用类对象
        
        NSLog(@"isInstance = %d",isInstance);//输出是1(ani本来就是Animal的实例对象,自然为1)
        
        
        Dog *dog=[Dog new];
        BOOL isInstance1=[dog isKindOfClass:[Animal class]];
        NSLog(@"isIstance1 = %d",isInstance1);//输出是1(Dog是Animal的子类,dog是Dog的实例对象,自然为1)
        
        
        Person *p=[Person new];
        BOOL isInstance2=[p isKindOfClass:[Animal class]];
        NSLog(@"isInstance2 = %d",isInstance2);//输出是0(Person是无关的类,不是Animal的子类,p是Person的实例对象,所以为0)
        
        //*******************************************************
        
//        2)判断某个对象是否是该类的实例对象(单指这个类,不包括其子类)
        
//        isMemberOfClass使用格式:   [对象 isMemberOfClass:类对象];
       
        BOOL isInstance3=[ani isMemberOfClass:[Animal class]];
        NSLog(@"isInstance3 = %d",isInstance3);//输出是1(ani是Animal的实例对象,自然为1)

        BOOL isInstance4=[dog isMemberOfClass:[Animal class]];
        NSLog(@"isInstance4 = %d",isInstance4);//输出是0(dog是Dog的实例对象,Dog是Animal的子类,所以为0)
        
        //*******************************************************
        
//        3)判断某个类是不是另一个类的子类
        
//        isSubclassOfClass使用格式:    [类名/类对象 isSubclassOfClass:类对象];

        BOOL isInstance5=[Dog isSubclassOfClass:[Animal class]];
        NSLog(@"isInstance5 = %d",isInstance5);//输出为1(Dog是Animal的子类)

        BOOL isInstance6=[Animal isSubclassOfClass:[Dog class]];
        NSLog(@"isInstance6 = %d",isInstance6);//输出为0(Animal是Dog的父类)

        BOOL isInstance7=[[Dog class] isSubclassOfClass:[Animal class]];
        NSLog(@"isInstance7 = %d",isInstance7);//输出为1(这里注意前面可以写作类对象的形式)

//        BOOL isInstance8=[[Dog class] isSubclassOfClass:Animal];//这句话编译不会通过,因为后面Animal不是类对象的格式
        
        //*******************************************************
        
//        4)判断对象能否响应指定的方法
        
//        respondsToSelector使用格式:   [对象 respondsToSelector:方法的SEL];
//        BOOL isRespond1=ani respondsToSelector:<#(SEL)#>//这里显然参数传进来的应该是一个SEL的类型
//        我们一起来回顾一下SEL是什么。首先SEL表示方法的存储位置,我们一般先将方法包装为SEL类型,然后根据sel数据找到方法的地址,然后根据方法地址调用相应的方法。所以接下来我们应该这样处理:
        
        SEL s1=@selector(eat);//先将数据封装成SEL类型,获得方法的地址
        BOOL isRespond1=[dog respondsToSelector:s1];//eat是Dog里面的方法,dog是Dog的实例对象,所以可以访问
        NSLog(@"isRespond1 = %d",isRespond1);//输出为1
        
        SEL s2=@selector(bark);
        BOOL isRespond2=[ani respondsToSelector:s2];//ani是Animal的实例对象,但是bark是Dog的特有方法,所以说无法访问
        NSLog(@"isRespond = %d",isRespond2);//输出为0
        
//        所以说我们一般用实例对象调用方法之前,可以先作一个判断,如下:
//        if(isRespond1)
//        {
//            [dog eat];
//        }
//        else
//        {
//            NSLog(@"无法调用");
//        }
//        这样可以把错误扼杀在编译时,而不是到了运行的时候错误才发现。
        
        //*******************************************************
        
//        5)判断类能否调用(相应)指定的方法
        BOOL isRespond3=[Dog instancesRespondToSelector:s1];//s1是eat的SEL封装,eat是Dog的方法,所以可以调用
        NSLog(@"isRespond3 = %d",isRespond3);//输出为1
        
        BOOL isRespond4=[Animal instancesRespondToSelector:s1];//显然Animal不能调用其子类特有的方法
        NSLog(@"isRespond4 = %d",isRespond4);//输出为0
        
        SEL s3=@selector(abc);//abc方法是Animal中的
        BOOL isRespond5=[Dog instancesRespondToSelector:s3];//子类继承父类的abc方法,自然子类可以调用(★这里我要说明一点,如果你在父类中只写了abc方法的声明,而没有写实现的话,这里结果就是0,表示无法调用★)
        NSLog(@"isRespond5 = %d",isRespond5);//输出为1
        
        //*******************************************************
        
    }
    return 0;
}


———————————————————————————————————————————
响应方法(属于动态类型检测部分)

//响应方法(调用方法)
//1.响应无参方法
//2.响应有一个参数的方法
//3.响应有两个参数的方法(因为OC只提供了最多两个参数的响应方法)


#import <Foundation/Foundation.h>


@interface Animal : NSObject
-(void)run;
@end

@implementation Animal
-(void)run
{
    NSLog(@"Animal run!");
}
@end

@interface Dog : Animal
-(void)eat:(NSString *)foodName;
-(void)eat:(NSString *)foodName andDogName:(NSString *)dogName;
@end

@implementation Dog
-(void)eat:(NSString *)foodName
{
    NSLog(@"Dog eat %@",foodName);
}
-(void)eat:(NSString *)foodName andDogName:(NSString *)dogName
{
    NSLog(@"%@ eat %@",dogName,foodName);
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
//        以后调用方法,大都以这种形式调用
//        调用方法的时候,我们应该具备这样的思路,养成书写习惯:
//        ①首先调用方法得有 实例对象,所以要先创建实例对象
//        ②其次调用方法前要判断是否能调用,所以要有判断语句(对象是否能调用方法)
//        ③因为判定对象能调用方法的时候这个方法是以SEL的格式去调用的,所以先要将此方法转化成SEL的形式
//        ④然后判断结束后,返回值是1就调用,返回值是0就输出无法调用
//        ⑤最后调用 无参/有参 方法(本节所讲内容)
        //*******************************************************
        Animal *ani =[[Animal alloc]init];//①
        SEL s1=@selector(run);//③
        BOOL isRespond=[ani respondsToSelector:s1];//②
        if (isRespond) {//④
            [ani performSelector:s1];//⑤      实例对象调用无参方法
        }
        else
        {
            NSLog(@"无法调用!");
        }
        //*******************************************************
        Dog *dog=[[Dog alloc]init];
        SEL s2=@selector(eat:);//这里获取含多个参数的方法地址的时候,只写方法名即可
        BOOL isRespond2=[dog respondsToSelector:s2];
        if (isRespond2) {
            [dog performSelector:s2 withObject:@"coffee"];//实例对象调用含有一个参数的方法
        }
        else
        {
            NSLog(@"无法调用!");
        }
        //*******************************************************
        Dog *dog2=[[Dog alloc]init];
        SEL s3=@selector(eat:andDogName:);//只写方法名
        BOOL isRespond3=[dog respondsToSelector:s3];
        if (isRespond3) {
            [dog performSelector:s3 withObject:@"hotdog" withObject:@"bigmax"];//实例对象调用含两个参数的方法
        }
        else
        {
            NSLog(@"无法调用!");
        }
        //*******************************************************
    }
    return 0;
}


———————————————————————————————————————————