解决performSelector afterDelay在子线程不调用问题

有时候你会发现performSelector延时不起作用,啥原因呢
先看下面的例子

- (void)viewDidLoad {
    [super viewDidLoad];
    NSThread * thread = [[NSThread alloc] initWithTarget:self selector:@selector(threadRun) object:nil];
    [thread setName:@"com.will.thread"];
    [thread start];
}

- (void)threadRun{
    [self performSelector:@selector(runTime1) withObject:nil];
    [self performSelector:@selector(runTime2) withObject:nil afterDelay:1];
    [self performSelector:@selector(runTime2) withObject:nil afterDelay:1 inModes:@[NSDefaultRunLoopMode]];
}

- (void)runTime1{
    NSLog(@"xxoo1");
}

- (void)runTime2{
    NSLog(@"xxoo2");
}
image.png

为什么延时函数没有调用,因为你在子线程里调用延时函数,需要定时器,而子线程不同于主线程不会自动创建runloop,导致定时器没有工作,
解决方法有四种:

  • 在子线程里启动runloop
- (void)threadRun{    
    [self performSelector:@selector(runTime2) withObject:nil afterDelay:1];
    NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
    [runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
    [runLoop run];
}
  • performSelector设定主线程,不过没法设置延时函数
- (void)threadRun{
    [self performSelector:@selector(runTime2) onThread:[NSThread mainThread] withObject:nil waitUntilDone:NO];
    //或者:
    [self performSelectorOnMainThread:@selector(runTime2) withObject:nil waitUntilDone:NO];
}
  • 专门设定在已经开启了runloop的线程
- (void)threadRun{
    [self performSelector:@selector(runTime2) onThread:[[self class] _networkThread] withObject:nil waitUntilDone:NO modes:@[NSDefaultRunLoopMode]];
}

+ (void)_addRunLoop:(NSThread *)thread{
    @autoreleasepool {
        [[NSThread currentThread] setName:@"com.will.webimage.request"];
        NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
        [runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
        [runLoop run];
    }
}

+ (NSThread *)_networkThread{
    NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(_addRunLoop:) object:nil];
    [NSThread sleepForTimeInterval:1];
    [thread start];
    return thread;
}
  • 直接用GCD的延时方法
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 1*NSEC_PER_SEC), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    [self runTime2];
});

注意:

  • performSelector若不指定mode([self performSelector:@selector(method) withObject:nil];),默认自动创建mode为NSDefaultRunLoopMode

  • performSelector:onThread:withObject:waitUntilDone里waitUntilDone为NO,则会等待当前线程执行完,再执行selector里的方法

- (void)threadRun{    
    [self performSelector:@selector(runTime2) onThread:[NSThread currentThread] withObject:nil waitUntilDone:NO];
    
    NSLog(@"com.will.thread执行完毕");
    NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
    [runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
    [runLoop run];
}

打印如下


image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一、什么是runloop 字面意思是“消息循环、运行循环”。它不是线程,但它和线程息息相关。一般来讲,一个线程一次...
    WeiHing阅读 8,192评论 11 111
  • ======================= 前言 RunLoop 是 iOS 和 OSX 开发中非常基础的一个...
    i憬铭阅读 920评论 0 4
  • runtime 和 runloop 作为一个程序员进阶是必须的,也是非常重要的, 在面试过程中是经常会被问到的, ...
    made_China阅读 1,235评论 0 7
  • 一、RunLoop 的概念 字面意思是“消息循环、运行循环”,RunLoop 实际上就是一个事件循环对象,这个对象...
    风轻鱼蛋阅读 500评论 0 1
  • 风说, 你伤透了我的心, 我温柔的撩起你额头的发, 为你吹落一地桃花。 你在阳光下开心的笑, 那是我见过的最纯真的...
    麦草店主阅读 133评论 0 0