performSelector和respondsToSelector用法

一、performSelector调用和直接调用区别

下面两段代码都在主线程中运行,我们在看别人代码时会发现有时会直接调用,有时会利用performSelector调用,今天看到有人在问这个问题,我便做一下总结,

[delegate imageDownloader:self didFinishWithImage:image];

[delegate performSelector:@selector(imageDownloader:didFinishWithImage:)withObject:self withObject:image];

1、performSelector是运行时系统负责去找方法的,在编译时候不做任何校验;如果直接调用编译是会自动校验。如果imageDownloader:didFinishWithImage:image:不存在,那么直接调用 在编译时候就能够发现(借助Xcode可以写完就发现),但是使用performSelector的话一定是在运行时候才能发现(此时程序崩溃);Cocoa支持在运行时向某个类添加方法,即方法编译时不存在,但是运行时候存在,这时候必然需要使用performSelector去调用。所以有时候如果使用了performSelector,为了程序的健壮性,会使用检查方法- (BOOL)respondsToSelector:(SEL)aSelector;

2、直接调用方法时候,一定要在头文件中声明该方法的使用,也要将头文件import进来。而使用performSelector时候,可以不用import头文件包含方法的对象,直接用performSelector调用即可。

.person.m文件的实现内容

- (void)invoke {

    //1.使用perform调用不带参数的方法
    //SEL是一个类型:方法选择器  @selector(方法名)

    [self performSelector:@selector(demo1)];
    
    //2.使用perform调用带有一个参数的方法

    [self performSelector:@selector(demo2:) withObject:@"这是一个参数"];
    
    //3.使用perform调用带有两个个参数的方法

    [self performSelector:@selector(demo3:withName:) withObject:@"参数1" withObject:@"参数2"];
   
    
    //4.延迟调用某一个方法
    //代码不会停留在这里,这个任务会挂载后台,后面代码继续执行

    [self performSelector:@selector(demo1) withObject:NULL afterDelay:2.5];
    
}

//不带参数
- (void)demo1 {

    NSLog(@"这是demo1");
}

//带有一个参数的方法
- (void)demo2:(NSString *)str {
    NSLog(@"这是demo2 str:%@",str);
}

//带有两个参数的方法
- (void)demo3:(NSString *)str withName:(NSString *)name {

    NSLog(@"这是demo3,str:%@  name:%@",str,name);
}


//私有方法
- (void)testMethod {

    NSLog(@"这是一个私有方法");
}

三.perform调用的演示

     Person *p = [[Person alloc]init];

    [p invoke];

//  [p testMethod];会报错
    //使用perform调用私有方法
    [p performSelector:@selector(testMethod)];
    
    //判断某一个对象是否响应了某一个方法
    BOOL isRespond = [p respondsToSelector:@selector(testMethod)];
    if (isRespond) {
        [p performSelector:@selector(testMethod)];
    }else {
        NSLog(@"p没有实现这个方法");
    }
原文地址:https://www.cnblogs.com/wangyibo-666/p/5118935.html