IOS開(kāi)發(fā)中延遲執(zhí)行和取消
在 Objective-C 中延遲執(zhí)行還是很常見(jiàn)的需求,通常有如下幾種方式可供選擇:
performSelector:
想要延遲調(diào)用某個(gè)方法:
[self performSelector:@selector(delay) withObject:nil afterDelay:3.0];
取消延遲的方法:
[NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(delay) object:nil];
這里需要注意參數(shù)需要保持一致,否則取消失敗。
NSTimer
想要延遲調(diào)用某個(gè)方法:
self.timer = [NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(delay) userInfo:nil repeats:NO];
取消延遲的方法:
[self.timer invalidate];
GCD
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// ...
});
dispatch_after 是比較常用的方法,但是 Objective-C 中并沒(méi)有提供取消執(zhí)行的相關(guān) API。我們只能自己實(shí)現(xiàn)這個(gè)取消的邏輯:
typedef void (^Task)(BOOL cancel);
Task delay(NSTimeInterval time,void (^task)()) {
__block void (^closure)() = task;
__block Task result;
Task delayedClosure = ^(BOOL cancel){
if (closure) {
void (^internalClosure)() = closure;
if (!cancel) {
dispatch_async(dispatch_get_main_queue(), internalClosure);
}
}
closure = nil;
result = nil;
};
result = delayedClosure;
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(time * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
if (delayedClosure) {
delayedClosure(NO);
}
});
return result;
}
使用的話可以這樣:
delay(60, ^{
// ...
});
如果想要延遲,可以先聲明成成員變量并賦值:
@property (copy, nonatomic) Task task;
self.task = delay(60, ^{
// ...
});
最后在需要的地方取消就行:
self.task(YES);
這種寫(xiě)法的核心思想是根據(jù)傳入的 Bool 值,來(lái)控制 dispatch_after 回調(diào) block 中的方法是否需要執(zhí)行。看起來(lái)是取消了,但實(shí)際上還是被 GCD 放到 RunLoop 里去占用主線程資源了。
dispatch_source
我們還可以利用 dispatch_source 中的定時(shí)器,來(lái)實(shí)現(xiàn)延時(shí)/取消操作:
@property (strong, nonatomic) dispatch_source_t timer;
// 隊(duì)列
dispatch_queue_t queue = dispatch_get_main_queue();
// 創(chuàng)建 dispatch_source
dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
// 聲明成員變量
self.timer = timer;
// 設(shè)置兩秒后觸發(fā)
dispatch_time_t startTime = dispatch_time(DISPATCH_TIME_NOW, 3.0 * NSEC_PER_SEC);
// 設(shè)置下次觸發(fā)事件為 DISPATCH_TIME_FOREVER
dispatch_time_t nextTime = DISPATCH_TIME_FOREVER;
// 設(shè)置精確度
dispatch_time_t leeway = 0.1 * NSEC_PER_SEC;
// 配置時(shí)間
dispatch_source_set_timer(timer, startTime, nextTime, leeway);
// 回調(diào)
dispatch_source_set_event_handler(timer, ^{
// ...
});
// 激活
dispatch_resume(timer);
需要取消的話:
dispatch_source_cancel(self.timer);
相關(guān)文章
iOS Objective-c實(shí)現(xiàn)左右滑動(dòng)切換頁(yè)面
這篇文章主要為大家詳細(xì)介紹了iOS Objective-c實(shí)現(xiàn)左右滑動(dòng)切換頁(yè)面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08
IOS圖片無(wú)限輪播器的實(shí)現(xiàn)原理
這篇文章主要介紹了IOS圖片無(wú)限輪播器的實(shí)現(xiàn)原理的相關(guān)資料,需要的朋友可以參考下2016-03-03
iOS 基于AFNetworking下自簽名證書(shū)配置的方法
本篇文章主要介紹了iOS 基于AFNetworking下自簽名證書(shū)配置的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
使用objc runtime實(shí)現(xiàn)iOS閉環(huán)的懶加載功能
利用objc runtime的動(dòng)態(tài)性實(shí)現(xiàn)懶加載可以實(shí)現(xiàn)即可增加又可刪除功能,也可以避免污染類(lèi)型。這篇文章主要介紹了使用objc runtime實(shí)現(xiàn)iOS閉環(huán)的懶加載功能,需要的朋友可以參考下2019-06-06
ios動(dòng)態(tài)庫(kù)和靜態(tài)庫(kù)的區(qū)別
這篇文章主要介紹了ios動(dòng)態(tài)庫(kù)和靜態(tài)庫(kù)的區(qū)別,幫助大家更好的理解和學(xué)習(xí)使用ios開(kāi)發(fā),感興趣的朋友可以了解下2021-04-04
iOS NSURLSessionDownloadTask設(shè)置代理文件下載的示例
本篇文章主要介紹了iOS NSURLSessionDownloadTask設(shè)置代理文件下載的示例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
詳解iOS11關(guān)于導(dǎo)航欄問(wèn)題
本篇文章主要介紹了詳解iOS11關(guān)于導(dǎo)航欄問(wèn)題,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10

