iOS_performSelector:withObject:afterDelay:
创始人
2025-05-30 17:18:55

看一下以下方法的执行结果:

例1:全局并发队列中执行异步任务

dispatch_async(dispatch_get_global_queue(0, 0), ^{NSLog(@"1");[self performSelector:@selector(completion) withObject:nil afterDelay:0];NSLog(@"2");
});
- (void)completion {NSLog(@"3");
}
// 1 2

原因是:往全局并发队列里加入了异步任务,会创建子线程。而子线程默认不开启 runloop,而 afterDelay 是通过 NSTimer 实现的,NSTimer 又依赖 rumloop 实现。所以 导致 completion 无法执行。


例2:把 async 改成 sync :

dispatch_sync(dispatch_get_global_queue(0, 0), ^{NSLog(@"1");[self performSelector:@selector(completion) withObject:nil afterDelay:0];NSLog(@"2");
});
- (void)completion {NSLog(@"3");
}
// 1 2 3

原因是:往全局并发队列里加入了同步任务,sync 会等下次 runloop 循环时在主线程调用。


看一下官方文档说明:
请添加图片描述
这个方法会在当前线程的 runloop 中创建一个计时器来执行该消息。
该计时器默认在 runloop 的 default 模式(NSDefaultRunLoopMode)下执行。
当计时器触发时,线程尝试从 runloop 中出列该消息并执行该函数。
如果 runloop 在运行并且处于 default 模式则执行成功。否则计时器会一直等到 runloop 处于 default 模式。

如果希望消息能在其他模式下也能出列,可以使用performSelector:withObject:afterDelay:inModes: 代替。如:

[self performSelector:@selector(test) withObject:nil afterDelay:2 inModes:@[NSRunLoopCommonModes]];

如果不确定当前线程是否会主线程,可以使用performSelectorOnMainThread:withObject:waitUntilDone:performSelectorOnMainThread:withObject:waitUntilDone:modes: 方法确保函数在主线程上执行。
可以使用 cancelPreviousPerformRequestsWithTarget:cancelPreviousPerformRequestsWithTarget:selector:object: 取消排列等候消息。如:

[NSObject cancelPreviousPerformRequestsWithTarget:self];
[NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(completion) object:nil];

解决方案1:用 dispatch_after 代替

NSLog(@"1");
[self performSelector:@selector(completion) withObject:nil afterDelay:0];
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{NSLog(@"3");
});
NSLog(@"2");// 1 2 3

解决方案2:获取子线程 runloop

dispatch_async(dispatch_get_global_queue(0, 0), ^{NSLog(@"1");[self performSelector:@selector(completion) withObject:nil afterDelay:0];[[NSRunLoop currentRunLoop] run];NSLog(@"2");
});
- (void)completion {NSLog(@"3");
}
// 1 3 2

缺陷:会阻塞子线程

相关内容

热门资讯

宁武县召开2025 年安委会第... 来源: 宁武县融媒体中心 12月8日,宁武县召开2025 年安委会第五次全体(扩大)会暨全...
【理财小讲堂】一文带你读懂理财... 之前为大家介绍过,理财产品的资金主要投向三类资产,分别为:债权类资产、权益类资产以及商品及金融衍生品...
昊海生科拟受让瑞济生物19.8... 北京商报讯(记者 丁宁)12月12日晚间,昊海生科(688366)发布公告称,公司拟以自有资金383...
原创 飞... 散瓶批发参考价跌至1485元/瓶,原箱产品报价1495元,较年初价格累计跌幅超30%,曾经一瓶难求的...
王源北京跨晚活动展现独特氛围 近日,王源在北京举办的跨晚活动引发了广泛关注,现场氛围热烈,吸引了众多粉丝的参与。活动于2025年举...