GCD

-GCD中的核心词是dispatch queue。一个队列实际上就是一系列的代码块,这些代码可以在主线程或后台线程中以同步或者异步的方式执行。一旦队列创建完成,操作系统就接管了这个队列,并将其分配到任意一个核心中进行处理。不管有多少个队列,它们都能被系统正确地管理,这些都不需要开发者进行手动管理。队列遵循 FIFO 模式(先进先出),这意味着先进队列的任务会先被执行(想像在柜台前排队的队伍,排在第一个的会首先被服务,排在最后的就会最后被服务)。

  • 另一个重要的概念就是WorkItem(任务项)。一个任务项就是一个代码块,它可以随同队列的创建一起被创建,也可以被封装起来,然后在之后的代码中进行复用。正如你所想,任务项的代码就是 dispatch queue 将会执行的代码。队列中的任务项也是遵循 FIFO 模式。这些执行可以是同步的,也可以是异步的。对于同步的情况下,应用会一直堵塞当前线程,直到这段代码执行完成。而当异步执行的时候,应用先执行任务项,不等待执行结束,立即返回。
  • 在为主队列添加任务时,无论何时都要加倍小心。这个队列要随时用于界面响应以及用户交互。并且记住一点,所有与用户界面相关的更新都必须在主线程执行。如果你尝试在后台线程更新 UI,系统并不保证这个更新何时会发生,大多数情况下,这会都用户带来不好的体验。但是,所有发生在界面更新前的任务都可以在后台线程执行。
  • 我们不一定需要每次都创建自己的队列。系统维护的全局队列可以用来执行任何我们想执行的任务。至于队列在哪一个线程运行,iOS 维护了一个线程池,即一系列除主线程之外的线程,系统会从中挑选一至多条线程来使用(取决于你所创建的队列的数据,以及队列创建的方式)。哪一条线程会被使用,对于开发者来说是未知的,而是由系统根据当前的并发任务,处理器的负载等情况来进行“决定”。

Dispatch Queue

在 swift 中 dispatch queue 对应的类为DispatchQueue ,可以使用下面方法进行初始化

let queue = (label: String,
  qos: DispatchQoS = default,
  attributes: DispatchQueue.Attributes = default, 
  autoreleaseFrequency: DispatchQueue.AutoreleaseFrequency = default,
 target: DispatchQueue? = default)

在 Swift 3 当中,创建一个dispatch queue的最简单方式:

let queue = DispatchQueue(label: "com.appcode.myqueue")

queue.sync {
      print("同步执行")
 }
    queue.async {
      print("异步执行")
 }

Quality Of Service (QOS) 和优先级

  • 在使用GCDdispatch queue时,我们经常需要告诉系统,应用程序中的哪些任务比较重要,需要更高的优先级去执行。当然,由于主队列总是用来处理 UI 以及界面的响应,所以在主线程执行的任务永远都有最高的优先级。不管在哪种情况下,只要告诉系统必要的信息,iOS 就会根据你的需求安排好队列的优先级以及它们所需要的资源(比如说所需的 CPU 执行时间)。虽然所有的任务最终都会完成,但是,重要的区别在于哪些任务更快完成,哪些任务完成得更晚。

  • 用于指定任务重要程度以及优先级的信息,在 GCD 中被称为 Quality of Service (Qos)。事实上,Qos 是有几个特定值的枚举类型,我们可以根据需要的优先级,使用合适的 Qos 值来初始化队列。如果没有指定 Qos,则队列会使用默认优先级进行初始化。

/// 由低到高 优先级
 public static let unspecified: DispatchQoS

 public static let background: DispatchQoS

 public static let utility: DispatchQoS

 public static let `default`: DispatchQoS

 public static let userInitiated: DispatchQoS

 public static let userInteractive: DispatchQoS

并发队列


///public static let concurrent: DispatchQueue.Attributes
///public static let initiallyInactive: DispatchQueue.Attributes
///attributes 参数也可以接受另一个名为 initiallyInactive 的值。如果使用这个值,
///任务不会被自动执行,而是需要开发者手动去触发。
let anotherQueue = DispatchQueue(label:"com.appcode.anotherQueue", 
qos: .utility, attributes: .concurrent)

  • DispatchQueue 类的 activate() 方法会让任务开始执行。注意,这个队列并没有被指定为并发队列,因此它们会以串行的方式执行。
let anotherQueue = DispatchQueue(label:"com.appcode.anotherQueue",
 qos: .utility, attributes: .initiallyInactive)
anotherQueue.activate()
  • 现在的问题是,我们如何在指定initiallyInactive 的同时将队列指定为并发队列?其实很简单,我们可以将两个值放入一个数组当中,作为attributes的参数,替代原本指定的单一数值。
let anotherQueue = DispatchQueue(label:"com.appcode.anotherQueue", 
qos: .userInitiated, attributes: [.concurrent, .initiallyInactive])

延迟执行

let delayQueue = DispatchQueue(label: "com.appcoda.delayqueue", qos: .userInitiated)

print(Date())

let additionalTime: DispatchTimeInterval = .seconds(2)
delayQueue.asyncAfter(deadline: .now() + additionalTime) {
    print(Date())
}

全局队列和主队列


let globalQueue = DispatchQueue.global(qos: .userInitiated)

DispatchQueue.main.async {
    // Do something
}

DispatchWorkItem

-DispatchWorkItem 是一个代码块,它可以在任意一个队列上被调用,因此它里面的代码可以在后台运行,也可以在主线程运行。它的使用真的很简单,就是一堆可以直接调用的代码,而不用像之前一样每次都写一个代码块。

var value = 10
 let workItem = DispatchWorkItem {
       value += 5
 }
workItem.perform()/// 这行代码会在主线程上面调用

///其他队列执行
let queue = DispatchQueue.global()
queue.async {
    workItem.perform()
}

queue.async(execute: workItem)

  • 当一个任务项被调用后,你可以通知主队列(或者任何其它你想要的队列)
workItem.notify(queue: DispatchQueue.main) {
    print("value = ", value)
}


    var value = 10

    let workItem = DispatchWorkItem {
        value += 5
    }

    workItem.perform()

    let queue = DispatchQueue.global(qos: .utility)

    queue.async(execute: workItem)

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

推荐阅读更多精彩内容

  • GCD(Grand Central Dispatch) 介绍 GCD 属于系统级的线程管理,在 Dispatch ...
    fuyoufang阅读 4,606评论 0 10
  • 尽管 Grand Central Dispatch (GCD)已经存在一段时间了,但并非每个人都知道怎么使用它。这...
    coderFamer阅读 7,354评论 1 16
  • 基于自 raywenderlich.com 在2015年的两篇文章 Grand Central Dispatch ...
    seedante阅读 1,361评论 0 7
  • 一、本周计划(一)这周就要跑马,周二进行半马练习,每天拿出时间规划准备;(二)幸福双翼跟上进程,录制音频;、(三)...
    舜间永恒阅读 146评论 0 1
  • 很久没有遛弯了。 最近心情总是起起落落,心里那股子叛逆也上来了。 对啊,我就是不想考好,不想去迎合应试教育。 我呸...
    花房姑娘__阅读 219评论 0 0