美文网首页
iOS异步任务调度 - OperationQueue、Dispa

iOS异步任务调度 - OperationQueue、Dispa

作者: chernyog | 来源:发表于2020-05-22 10:36 被阅读0次

需求

有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

相关文章

  • iOS异步任务调度 - OperationQueue、Dispa

    需求 有A、B、C、D、E、F 六个任务,D依赖A、B的结果,E依赖B、C的结果,F依赖D、E的结果,A、B、C互...

  • Spring定时任务(串行/并行)以及异步任务

    一。Spring提供了两种调度任务的方式。 调度任务,@Scheduled异步任务,@Async 1)spring...

  • iOS 主队列 全局队列的区别

    主队列(串行):主要负责调度主线程度的任务, 不论在异步任务还是同步任务都不能开辟新的线程 主队列异步任务将异步任...

  • iOS 控制任务执行顺序 OperationQueue

    控制任务执行的顺序 github地址YSSimpleOperation 情景 任务A依赖于任务a,b,c的完成后才...

  • 构造异步NSOperation进行异步任务调度

    开发中经常遇到异步任务之间有依赖关系,需要对执行顺序进行调度的情况。 比如,一个页面要组合多个后端接口的数据,必须...

  • GCD 自锁与barrier

    GCD基础 GCD 是异步任务的技术之一,开发者可以用它将自定义的任务(task)追加到适当的派发队列(dispa...

  • Objective-C 带有 block 的 NSTimer

    有的时候 NSTimer 的调度任务比较复杂,需要在调度方法中执行异步操作,并且在异步操作中需要回调原方法中的回调...

  • Dispatch Queue

    Grend Central Dispatch(GCD)调度队列是执行任务的强大工具。调度队列让我们可以与调用者异步...

  • 基于红黑树的TreeMap使用

    背景 最近在项目中做异步任务调度服务的时候,用到红黑树来实现异步任务的管理,挑选出最符合条件的任务执行,于是使用到...

  • iOS调度队列

    GCD调度队列是执行任务的强大工具。调度队列允许您相对于调度者异步或者同步的执行任意代码块。您能够使用调度队列来执...

网友评论

      本文标题:iOS异步任务调度 - OperationQueue、Dispa

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