GCD延时任务取消

作者: 吴泉 | 来源:发表于2016-07-05 17:04 被阅读4912次

1.dispatch_block_cancel

iOS8之后可以调用dispatch_block_cancel来取消(需要注意必须用dispatch_block_create创建dispatch_block_t

let task2 = dispatch_block_create(DISPATCH_BLOCK_BARRIER, { print("output after 5 seconds") })
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, Int64(5 * Double(NSEC_PER_SEC))), dispatch_get_main_queue(), task2)
dispatch_block_cancel(task2)

2.Swift模拟取消

实际上delay时间到了之后还是会执行延时block,只是这时真正要执行的任务block已经为空

import Foundation

typealias Task = (cancel: Bool) -> Void

func gcdDelay(time: NSTimeInterval, task: ()->()) -> Task? {
    func dispatch_later(block: ()->()) {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, Int64(time * Double(NSEC_PER_SEC))), dispatch_get_main_queue(), block)
    }
    
    var closure: dispatch_block_t? = task
    var result: Task?
    
    let delayedClosure: Task = {
        cancel in
        if let internalClosure = closure {
            if cancel == false {
                dispatch_async(dispatch_get_main_queue(), internalClosure)
            }
        }
        closure = nil
        result = nil
    }
    
    result = delayedClosure
    
    dispatch_later { 
        if let delayedClosure = result {
            delayedClosure(cancel: false)
        }
    }
    
    return result
}

func gcdCancel(task: Task?) {
    task?(cancel: true)
}

func gcdTest(){
    let task = gcdDelay(5) {
        print("output after 5 seconds")
    }
    gcdCancel(task)
}

3.Objective-C模拟取消

原理与2类似
GCDDelay.h

#import <Foundation/Foundation.h>

typedef void(^GCDTask)(BOOL cancel);
typedef void(^gcdBlock)();

@interface GCDDelay : NSObject

+(GCDTask)gcdDelay:(NSTimeInterval)time task:(gcdBlock)block;
+(void)gcdCancel:(GCDTask)task;

+(void)gcdTest;

@end

GCDDelay.m

@implementation GCDDelay

+(void)gcdLater:(NSTimeInterval)time block:(gcdBlock)block
{
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(time * NSEC_PER_SEC)), dispatch_get_main_queue(), block);
}

+(GCDTask)gcdDelay:(NSTimeInterval)time task:(gcdBlock)block
{
    __block dispatch_block_t closure = block;
    __block GCDTask result;
    GCDTask delayedClosure = ^(BOOL cancel){
        if (closure) {
            if (!cancel) {
                dispatch_async(dispatch_get_main_queue(), closure);
            }
        }
        closure = nil;
        result = nil;
    };
    result = delayedClosure;
    [self gcdLater:time block:^{
        if (result)
            result(NO);
    }];
    
    return result;
}

+(void)gcdCancel:(GCDTask)task
{
    task(YES);
}

+(void)gcdTest
{
    GCDTask task = [self gcdDelay:5 task:^{
        NSLog(@"oc output after 5 seconds");
    }];
    [self gcdCancel:task];
}

@end

参考:
1.swifter.tips之GCD和延时调用

相关文章

  • GCD延时任务取消

    1.dispatch_block_cancel iOS8之后可以调用dispatch_block_cancel来取...

  • 「取消」 GCD 延时队列

    在 iOS SDK 里面是无法取消提交的任务的,这里实现一个可以取消的例子。 void dispatch_afte...

  • GCD延时取消

    平常的开发中,经常要用到延时处理功能,一般比较实用的有2种方法,一是performSelector:withObj...

  • 【瞎搞iOS开发09】聊一聊Block和[Block copy]

    本文扯些啥 一种取消GCD延时任务的解决方案 非常简单的介绍3种Block类型 ARC模式打印引用计数retian...

  • 如何取消GCD任务

    1.dispatch_block_cancel iOS8之后可以调用dispatch_block_cancel来取...

  • 如何取消GCD任务

    转载自:教你如何取消GCD任务 GCD 是一种非常方便的使用多线程的方式。通过使用 GCD,我们可以在确保尽量简单...

  • 高级iOS面试题全纪录

    iOS基础: 多线程使用,gcd跟operation区别,怎么取消正在执行的gcd任务 GCD 系列知识总结 NS...

  • 关于NSTimer

    关于NSTimer 在工作中经常会做一些延时任务,或者周期性任务,有时候也需要对取消延时任务操作。 延时任务一般有...

  • iOS开发-延时执行

    1.GCD 注:dispatch_after方法本质是将任务延时添加到队列中,并不是延时执行任务,所以在对时间要求...

  • 多线程常见实例

    1. GCD的cancel操作 NSOperation 是基于GCD来实现的,其中的任务可以设置依赖、暂停、取消,...

网友评论

本文标题:GCD延时任务取消

本文链接:https://www.haomeiwen.com/subject/atghjttx.html