11--多线程05--GCD队列优先级

想了许久,还是决定将GCD的内容尽量写得简单一点,所以多分几个章节,每个章节内容尽量少。

一、如何创建一个队列

GCD提供了3个创建队列的API,下面就通过这是三个API来看看队列的优先级中的那些事。

1.1 dispatch_get_main_queue()

  • 测试代码:
dispatch_queue_t mainQueue = dispatch_get_main_queue();
  • 接受参数:void

  • 输出结果1:po mainQueue

(lldb) po mainQueue
<OS_dispatch_queue_main: com.apple.main-thread[0x1003364c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = com.apple.root.default-qos.overcommit[0x100336940], width = 0x1, state = 0x001ffe1000000304, in-flight = 0, thread = 0x307 }>
  1. OS_dispatch_queue_main: com.apple.main-thread[0x1003364c0]:主队列队列类型:OS_dispatch_queue_main,标签:com.apple.main-thread[0x1003364c0]:队列的地址;
  2. xref/ref/sref = 1:ref一般表示引用,只要是对象,都满足ARC里面的引用计数规则,所以这三个属性应该是表示某种引用计数。但这个1表示的是自己的引用计数,还是target的引用计数还无法确定,答案可能藏在后面的分析中;
  3. target = com.apple.root.default-qos.overcommit[0x100336940]:让人耳目一新的属性出来的,主队列仍然是依赖于某个队列,这个队列又有啥特点呢?
  4. width = 0x1:这个简单,串行队列,宽度为1,表示任务只能逐个执行;
  5. state = 0x001ffe1000000304:是一个地址,因为对内存地址的取值不太清楚😣,所以这里也无法确定是一个对象地址,还是表示一个值,希望后面的分析中还能看到;
  6. in-flight = 0:第一次见,这里也不做展开分析,重点是队列的优先级;
  7. thread = 0x307:众所周知,系统会给主队列分配一个主线程;
  • 输出结果2:
(lldb) po 0x100336940
<OS_dispatch_queue_global: com.apple.root.default-qos.overcommit[0x100336940] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>
  1. OS_dispatch_queue_global:主线程所依赖的队列是一个全局队列,有点意思😁;
  2. target:target为空,表示已经没有所依赖的队列了;

1.2 dispatch_get_global_queue()

  • 测试代码:

    // 全局队列(默认优先级)
    dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    NSLog(@"globalQueue : %@", globalQueue);
    // 全局队列(高优先级)
    dispatch_queue_t globalQueueHigh = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);
    NSLog(@"globalQueueHigh : %@", globalQueueHigh);
    // 全局队列(低优先级)
    dispatch_queue_t globalQueueLow = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0);
    NSLog(@"globalQueueLow : %@", globalQueueLow);
    
  • 接受参数:

    1. intptr_t identifier:队列优先级,只接受下面四个值:
    - DISPATCH_QUEUE_PRIORITY_HIGH:         QOS_CLASS_USER_INITIATED
    - DISPATCH_QUEUE_PRIORITY_DEFAULT:      QOS_CLASS_DEFAULT
    - DISPATCH_QUEUE_PRIORITY_LOW:          QOS_CLASS_UTILITY
    - DISPATCH_QUEUE_PRIORITY_BACKGROUND:   QOS_CLASS_BACKGROUND
    
    1. uintptr_t flags:一个预留参数,目前只接受0,其他任何值都会返回NULL;
  • 输出结果:

(lldb) po globalQueue
<OS_dispatch_queue_global: com.apple.root.default-qos[0x1003368c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po globalQueueHigh
<OS_dispatch_queue_global: com.apple.root.user-initiated-qos[0x1003369c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po globalQueueLow
<OS_dispatch_queue_global: com.apple.root.utility-qos[0x1003367c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>
  1. 这是三个不同的队列,而且地址是相邻的,间距128字节,可能是队列size,;
  2. 这三个队列都是OS_dispatch_queue_global类型的队列,而且target都为空,有可能是根队列。主队列所依赖的队列地址为:0x100336940,与0x1003367c0中间还有两个间距,将它们打印出来瞅瞅;
  • 输出结果2:
po 0x1003365c0
<OS_dispatch_queue_global: com.apple.root.maintenance-qos[0x1003365c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x1003366c0
<OS_dispatch_queue_global: com.apple.root.background-qos[0x1003366c0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>
  1. 结果不负众望,确实有两个队列,而且队列标签还不尽相同,分别是:com.apple.root.maintenance-qoscom.apple.root.background-qos
  2. 既然有规律,那就按照这个规律打印更多的地址试试看;
  • 输出结果3:
 po 0x100336Ac0
<OS_dispatch_queue_global: com.apple.root.user-interactive-qos[0x100336ac0] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336Bc0
4298337216

(lldb) po 0x100336Cc0
4298337472

(lldb) po 0x100336Dc0
4298337728

(lldb) po 0x100336Ec0
4298337984

(lldb) po 0x100336Fc0
4298338240
  1. 很可惜,只找到了地址:0x100336ac0所对应的队列,听说有12种,但目前只看到了6种,会不会是我们创建的队列不够多,而根队列的加载方式属于懒加载,所以才无法找到其他的几条队列呢?
  2. 这个问题就放在第二轮,创建更多类型的队列之后再验证。

1.3 dispatch_queue_create()

前面两个方法都是获取系统给我们提供的队列,GCD也提供了API给我们自己创建队列,上代码。

  • 测试代码:
// 并行队列(默认优先级)
dispatch_queue_t concurrentQueue = dispatch_queue_create("com.xy.concurrentQueue", DISPATCH_QUEUE_CONCURRENT);
NSLog(@"concurrentQueue : %@", concurrentQueue);
// 串行队列(默认优先级)
dispatch_queue_t serialQueue = dispatch_queue_create("com.xy.serialQueue", DISPATCH_QUEUE_SERIAL);
NSLog(@"serialQueue : %@", serialQueue);
  • 接受参数:

    1. label:队列的标签;
    2. dispatch_queue_attr_t attr:可以有两种值:
      • 预定义好的属性:DISPATCH_QUEUE_SERIALDISPATCH_QUEUE_CONCURRENT,分别表示串行和并行队列;
      • 通过dispatch_queue_attr_make_with_创建特定的属性
  • 输出结果:

(lldb) po serialQueue
<OS_dispatch_queue_serial: com.xy.serialQueue[0x100504490] = { xref = 1, ref = 1, sref = 1, target = com.apple.root.default-qos.overcommit[0x100336940], width = 0x1, state = 0x001ffe2000000000, in-flight = 0}>

(lldb) po concurrentQueue
<OS_dispatch_queue_concurrent: com.xy.concurrentQueue[0x100504200] = { xref = 1, ref = 1, sref = 1, target = com.apple.root.default-qos[0x1003368c0], width = 0xffe, state = 0x0000041000000000, in-flight = 0}>
  1. 串行队列serialQueue的目标队列0x100336940,跟主队列的目标队列是同一个;
  2. 并行队列concurrentQueue的目标队列0x1003368c0,就是我们上面定义的globalQueue,默认优先级的全局队列;

1.4 dispatch_queue_attr_t属性

对于dispatch_queue_create方法,除了接受两个预定义的类型,还接受dispatch_queue_attr_t类型,而GCD中提供了dispatch_queue_attr_t类型的三种构造方法,因为本章主要研究的是队列的优先级,所以采用:dispatch_queue_attr_make_with_qos_class方法来构造属性

dispatch_queue_attr_make_with_
  • 测试代码:
// 串行队列(指定优先级)
dispatch_queue_attr_t serialAttr = dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_SERIAL,
                                                                           QOS_CLASS_USER_INTERACTIVE,
                                                                           -1);
dispatch_queue_t userInteractiveQueue = dispatch_queue_create("com.xy.interactive.serialQueue", serialAttr);
NSLog(@"userInteractiveQueue : %@", userInteractiveQueue);
  • 接受参数:

    1. dispatch_queue_attr_t attr:看起来还是它本身,但从注释中给的例子、以及前面看到的两个预定义的参数来说,这里应该填:DISPATCH_QUEUE_SERIALDISPATCH_QUEUE_CONCURRENT
    2. dispatch_qos_class_t:注释中也明确的指出,它只接受5种类型,其他类型都会返回NULL
    @param qos_class
    A QOS class value:
     - QOS_CLASS_USER_INTERACTIVE
     - QOS_CLASS_USER_INITIATED
     - QOS_CLASS_DEFAULT
     - QOS_CLASS_UTILITY
     - QOS_CLASS_BACKGROUND
    Passing any other value is undefined.
    

    虽然知道它表示的是优先级,然而是怎么表示优先级的并不知道,那就继续探索,最后找到了这些枚举值的具体值的定义,有了大小,结合名称定义上来看,数值大的,优先级更高。

    __QOS_ENUM(qos_class, unsigned int,
        QOS_CLASS_USER_INTERACTIVE
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x21,
        QOS_CLASS_USER_INITIATED
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x19,
        QOS_CLASS_DEFAULT
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x15,
        QOS_CLASS_UTILITY
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x11,
        QOS_CLASS_BACKGROUND
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x09,
        QOS_CLASS_UNSPECIFIED
                __QOS_CLASS_AVAILABLE(macos(10.10), ios(8.0)) = 0x00,
    );
    

    细心观察以下数量,出去最后一种QOS_CLASS_UNSPECIFIED类型,还有5种类型,乘以2,也才10种,距离12种还差两种。

    1. int relative_priority:从命名上来看,可以理解为关联优先级,或者是辅助优先级判断。注释中给出的取值范围是:[0, QOS_MIN_RELATIVE_PRIORITY]#define QOS_MIN_RELATIVE_PRIORITY (-15),==> [0, -15]。第二个参数的优先级是针对系统定义的12种根队列中的某一种,而我们可以创建非常多依赖某一个根队列中的队列,那这些优先级相同的队列之间应该也要有个优先级关系,使用该属性区分就对了,后面在源码中也可以看到这个属性。
  • 输出结果:

po userInteractiveQueue
<OS_dispatch_queue_serial: com.xy.interactive.serialQueue[0x100605020] = { xref = 1, ref = 1, sref = 1, target = com.apple.root.user-interactive-qos.overcommit[0x100336b40], width = 0x1, state = 0x001ffe2000000000, in-flight = 0}>
  1. 一种新的类型:com.apple.root.user-interactive-qos.overcommit,而且这个地址,在前面打印的时候是没有的,现在有队列添加到上面的时候,地址中才有内容;
  2. 根队列的地址空间是固定的,尽管那个位置指定队列没有用到,也不会分配给其他对象使用;
  3. 根队列中有添加队列时,则会动态的激活该地址,使该固定地址指向固定的根队列;
  4. 这种数据结构看起来像数组;

1.5 验证更多类型的队列

  • 验证代码:
// 主队列(默认优先级)
    dispatch_queue_t mainQueue = dispatch_get_main_queue();

    // 全局队列(默认优先级)
    dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    
    // 全局队列(高优先级)
    dispatch_queue_t globalQueueHigh = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);
    
    // 全局队列(低优先级)
    dispatch_queue_t globalQueueLow = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0);
    
    // 并行队列(默认优先级)
    dispatch_queue_t concurrentQueue = dispatch_queue_create("com.xy.concurrentQueue", DISPATCH_QUEUE_CONCURRENT);
    
    // 串行队列(默认优先级)
    dispatch_queue_t serialQueue = dispatch_queue_create("com.xy.serialQueue", DISPATCH_QUEUE_SERIAL);
    
    // 串行队列(指定优先级)
    dispatch_queue_attr_t serialAttr = dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_SERIAL,
                                                                               QOS_CLASS_USER_INTERACTIVE,
                                                                               -1);
    dispatch_queue_t userInteractiveQueue = dispatch_queue_create("com.xy.userInitiatedQueue.serialQueue", serialAttr);
    
    dispatch_queue_attr_t userInitiatedAttr = dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_CONCURRENT,
                                                                               QOS_CLASS_USER_INITIATED,
                                                                               -1);
    dispatch_queue_t userInitiatedQueue = dispatch_queue_create("com.xy.userInitiatedQueue.serialQueue", userInitiatedAttr);
    
    dispatch_queue_attr_t utilityAttr = dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_SERIAL,
                                                                               QOS_CLASS_UTILITY,
                                                                               -1);
    dispatch_queue_t utilityQueue = dispatch_queue_create("com.xy.utilityQueue.serialQueue", utilityAttr);
  • 输出结果:
(lldb) po 0x100336540
<OS_dispatch_queue_mgr: com.apple.libdispatch-manager[0x100336540] = { xref = -2147483648, ref = -2147483648, sref = 1, target = com.apple.root.user-interactive-qos.overcommit[0x100336b40], width = 0x1, state = 0x001ffe1000000000, in-flight = 0}>

(lldb) po 0x100336640
<OS_dispatch_queue_global: com.apple.root.maintenance-qos.overcommit[0x100336640] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336740
<OS_dispatch_queue_global: com.apple.root.background-qos.overcommit[0x100336740] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336840
<OS_dispatch_queue_global: com.apple.root.utility-qos.overcommit[0x100336840] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336940
<OS_dispatch_queue_global: com.apple.root.default-qos.overcommit[0x100336940] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336a40
<OS_dispatch_queue_global: com.apple.root.user-initiated-qos.overcommit[0x100336a40] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336b40
<OS_dispatch_queue_global: com.apple.root.user-interactive-qos.overcommit[0x100336b40] = { xref = -2147483648, ref = -2147483648, sref = 1, target = [0x0], width = 0xfff, state = 0x0060000000000000, in-barrier}>

(lldb) po 0x100336c40
4298337344

(lldb) po 0x100336d40
4298337600

(lldb) po 0x100336e40
4298337856

(lldb) po 0x100336f40
4298338112
  1. 多了一种类型:com.apple.root.user-initiated-qos.overcommit
  2. 根队列之间的顺序是相对的,但位置不是绝对的。(推翻上面的猜测:根队列的地址空间是固定的

二、总结

2.1 获取队列的三种方式

  1. dispatch_get_main_queue()
  2. dispatch_get_global_queue()
  3. dispatch_queue_create()

2.2 队列的优先级

  1. 上面三种方法获取的队列都是默认优先级的;
  2. 可以通过dispatch_queue_attr_make_with_qos_class方法创建指定优先级的属性;
  3. 系统提供12种根队列,所有的非根队列都依赖其中一个根队列。可以参考为父视图与子视图之间的关系;
  4. __QOS_ENUM指定的优先级是target的优先级,表示你希望你的队列依赖系统的哪个优先级的队列;
  5. relative_priority表示队列在根队列中的优先级,取值范围[0,-15]。一个根队列上会有很多子队列,而这些子队列之间也存在优先级关系,就用这个属性标识。

2.3 更多疑问

  1. 是否可以将一个队列的target设置为自定义队列,而不是根队列?

当然可以。看系统API:dispatch_set_target_queue(dispatch_object_t, dispatch_queue_t)

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

推荐阅读更多精彩内容