iOS异步任务调度 - OperationQueue、DispatchGroup、DispatchSemaphore、ReactiveSwift

需求

有A、B、C、D、E、F 六个任务,D依赖A、B的结果,E依赖B、C的结果,F依赖D、E的结果,A、B、C互不依赖,D、F之间不依赖,可以并发,整体要尽快执行完成。

参考实现

一、通过 OperationQueue 添加依赖关系实现

/// 通过 OperationQueue 控制异步任务调度
func testOperation() {
    // 创建异步任务
    let operaA = BlockOperation {
        self.execTask("A")
    }
    let operaB = BlockOperation {
        self.execTask("B")
    }
    let operaC = BlockOperation {
        self.execTask("C")
    }
    let operaD = BlockOperation {
        self.execTask("D")
    }
    let operaE = BlockOperation {
        self.execTask("E")
    }
    let operaF = BlockOperation {
        self.execTask("F")
    }
    let operaG = BlockOperation {
        self.printf("\nFinished\n")
    }
    // 添加依赖关系
    operaD.addDependency(operaA)
    operaD.addDependency(operaB)
    operaE.addDependency(operaB)
    operaE.addDependency(operaC)
    operaF.addDependency(operaD)
    operaF.addDependency(operaE)
    operaG.addDependency(operaF)

    let queue = OperationQueue()
    // Adds the specified operation to the receiver.
    queue.addOperation(operaA)
    queue.addOperation(operaB)
    queue.addOperation(operaC)
    queue.addOperation(operaD)
    queue.addOperation(operaE)
    queue.addOperation(operaF)
    queue.addOperation(operaG)
    // 设置最大并发数
    queue.maxConcurrentOperationCount = queue.operations.count
}

二、通过 DispatchGroup 控制异步任务调度

/// 通过 DispatchGroup 控制异步任务调度
func testDispatchGroup() {
    // 创建DispatchGroup
    let groupD = DispatchGroup()  // A、B -> D
    let groupE = DispatchGroup()  // B、C -> E
    let groupF = DispatchGroup()  // D、E -> F
    let groupFinish = DispatchGroup()  // 完成
    // 创建异步任务
    let queueA = DispatchQueue(label: "A")
    let queueB = DispatchQueue(label: "B")
    let queueC = DispatchQueue(label: "C")
    let queueD = DispatchQueue(label: "D")
    let queueE = DispatchQueue(label: "E")
    let queueF = DispatchQueue(label: "F")

    groupD.enter()
    queueA.async {
        self.execTask("A")
        groupD.leave()
    }
    groupD.enter()
    groupE.enter()
    queueB.async {
        self.execTask("B")
        groupD.leave()
        groupE.leave()
    }
    groupE.enter()
    queueC.async {
        self.execTask("C")
        groupE.leave()
    }

    groupF.enter()
    groupD.notify(queue: DispatchQueue.global()) {
        // A、B任务已完成,开始D任务
        queueD.async {
            self.execTask("D")
            groupF.leave()
        }
    }
    groupF.enter()
    groupE.notify(queue: DispatchQueue.global()) {
        // B、C任务已完成,开始E任务
        queueE.async {
            self.execTask("E")
            groupF.leave()
        }
    }
    groupFinish.enter()
    groupF.notify(queue: DispatchQueue.global()) {
        // D、E任务已完成,开始F任务
        queueF.async {
            self.execTask("F")
            groupFinish.leave()
        }
    }
    groupFinish.notify(queue: DispatchQueue.global()) {
        self.printf("\nFinished\n")
    }
}

三、通过信号量控制异步任务调度

/// 通过信号量控制异步任务调度
func testDispatchSemaphore() {
    let value = 0
    let semaphoreD =  DispatchSemaphore(value: value)  // A、B -> D
    let semaphoreE =  DispatchSemaphore(value: value)  // B、C -> E
    let semaphoreF =  DispatchSemaphore(value: value)  // E、D -> F
    let semaphoreG =  DispatchSemaphore(value: value)  // 完成
    // 创建异步任务
    let queueA = DispatchQueue(label: "A")
    let queueB = DispatchQueue(label: "B")
    let queueC = DispatchQueue(label: "C")
    let queueD = DispatchQueue(label: "D")
    let queueE = DispatchQueue(label: "E")
    let queueF = DispatchQueue(label: "F")
    queueA.async {
        self.execTask("A")
    }
    queueB.async {
        self.execTask("B")
        semaphoreD.signal()
    }
    queueC.async {
        self.execTask("C")
        semaphoreE.signal()
    }
    semaphoreD.wait()
    queueD.async {
        Thread.sleep(forTimeInterval: 2)
        self.execTask("D")
    }
    semaphoreE.wait()
    queueE.async {
        Thread.sleep(forTimeInterval: 2)
        self.execTask("E")
        semaphoreF.signal()
    }
    semaphoreF.wait()
    queueF.async {
        self.execTask("F")
        semaphoreG.signal()
    }
    semaphoreG.wait()
    printf("\nFinished\n")
}

四、ReactiveSwift信号流

typealias SignalProducerHandler = SignalProducer<String, Error>
/// 通过ReactiveSwift信号流控制异步任务调度
private func testSignalProducer() {
    /// 创建异步任务
    let signalA = setupSignal("A")
    let signalB = setupSignal("B")
    let signalC = setupSignal("C")
    let signalD = setupSignal("D")
    let signalE = setupSignal("E")
    let signalF = setupSignal("F")
    /// A、B、C并发请求
    SignalProducer.merge(signalA, signalB, signalC).collect().on(
        failed: { error in
            self.printf("\(error)")
        }
    ).flatMap(.concat, { (_) -> SignalProducer<[String], Error> in
        // D、E并发请求
        return SignalProducer.merge(signalD, signalE).collect().on(
            failed: { error in
                self.printf("\(error)")
        }
        )
    }).flatMap(.concat) { (data) -> SignalProducerHandler in
        // 请求F
        return signalF
        }.on(
            failed: { error in
                self.printf("\(error)")
            },
            completed: {
                self.printf("\nFinished\n")
            },
            value: { data in
                self.printf("\ndata: \(data)\n")
            }
    ).start()
}

问题:

这里实现的是【A、B、C并发请求】,成功后再串联【D、E并发请求】的结果,最后再请求F,与题目中的“整体要尽快执行完成”描述不太相符,目前没有想到怎么控制请求B重复请求的办法,做标记还是用其他方法,待求解。。。


私有成员

private let _time = 3
// mock异步任务
private func execTask(_ taskName: String) {
    for _ in 0..<_time { printf(taskName) }
}
/// 打印log
private func printf(_ items: String) {
    print(items, terminator: "\t")
}
// 创建异步请求信号
private func setupSignal(_ taskName: String) -> SignalProducerHandler {
    return SignalProducerHandler() {[weak self] sink, _ in
        DispatchQueue(label: taskName).async {
            self?.execTask(taskName)
        }
        sink.send(value: taskName)
        sink.sendCompleted()
    }
}

参考log输出

A   A   A   C   C   C   B   B   B   E   D   D   D   E   E   F   F   F   
Finished

A   A   A   B   B   B   C   C   C   D   D   D   E   E   E   F   F   F   
Finished

B   C   A   A   A   C   C   B   B   D   E   D   E   E   D   F   F   F   
Finished
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352