美文网首页
iOS -dispatch_sync阻塞的是线程还是队列?

iOS -dispatch_sync阻塞的是线程还是队列?

作者: Pandakingli | 来源:发表于2020-04-13 14:27 被阅读0次


1.几个概念

1.1 进程--独立运转的公司

在iOS 中 一个进程就是一个正在运行的一个应用程序; 比如 QQ.app ,而且一个App只能有一个进程不像安卓支持多个进程。

进程就像一个独立运转的公司一样,
有自己的车间(内存空间)和库房(磁盘空间),有保安进行保护(别的App(进程)是不允许随意访问的)。

1.1线程--车间里的工人

线程是执行任务的,也就是干活的工人,给什么活儿就干什么活儿,任劳任怨。

线程是CPU调度的最小单元;
线程的作用:执行app的代码;
一个进程(App)至少有一个线程,这个进程叫做主线程;
一个线程必须有父进程
一个线程只能同时执行一个任务(一个工人同时只能干一个活儿)

1.2 队列--公司的生产任务分配列表

主队列:本质上是一个 串行队列,在iOS系统上主队列任务只会在主线程上执行

串行队列--这个生产流程必须按照顺序一个接一个完成

并行队列-这个生产流程可以分给几个工人同时加工来,最后装配完成

2. 关于dispatch_sync

2.1 第一个例子

int main(int argc, char * argv[]) {
    NSString * appDelegateClassName;
    @autoreleasepool {

        appDelegateClassName = NSStringFromClass([AppDelegate class]);
         NSLog(@"任务1");
        dispatch_queue_t gQ = dispatch_get_global_queue(0, 0);
        dispatch_queue_t mQ = dispatch_get_main_queue();
        dispatch_sync(mQ, ^(void){
            NSLog(@"xxx,当前线程=%@",[NSThread currentThread]);

               });
        NSLog(@"任务2");
    }
    return UIApplicationMain(argc, argv, nil, appDelegateClassName);
}

上面的代码执行结果是什么呢?
结果是打印完"任务1",程序就崩溃了。

2.1 第二个例子

int main(int argc, char * argv[]) {
    NSString * appDelegateClassName;
    @autoreleasepool {
        // Setup code that might create autoreleased objects goes here.
        appDelegateClassName = NSStringFromClass([AppDelegate class]);
         NSLog(@"任务1");
        
        dispatch_queue_t gQ = dispatch_get_global_queue(0, 0);
        dispatch_queue_t mQ = dispatch_get_main_queue();
        
        dispatch_queue_t cQ = dispatch_queue_create("cQueue", DISPATCH_QUEUE_SERIAL);
        
        dispatch_sync(cQ, ^(void){
            NSLog(@"xxx,当前线程=%@",[NSThread currentThread]);

               });
        NSLog(@"任务2");
    }
    return UIApplicationMain(argc, argv, nil, appDelegateClassName);
}
2020-04-13 12:27:47.154294+0800 xx[11263:5124299] 任务1
2020-04-13 12:27:47.154689+0800 xx[11263:5124299] xxx,当前线程=<NSThread: 0x2801e6ec0>{number = 1, name = main}
2020-04-13 12:27:47.154720+0800 xx[11263:5124299] 任务2

2.3 对比一下前两个例子

例子一中:
int main(int argc, char * argv[])是主队列的任务,要在主线程里面执行。
这样如果在执行main()任务的时候,向主队列添加新的任务并且同步执行。会导致新的任务main()任务形成等待循环,导致崩溃。

例子一

例子二中:
int main(int argc, char * argv[])是在主队列、主线程里面执行的。
这样如果在执行main()任务的时候,向自定义串行队列cQ添加新的任务并且同步执行。此时main()任务被阻塞,等待cQ内的任务完成,cQ在主线程执行完任务后,main()任务得以继续执行。

例子二

所以:dispatch_sync并不是阻塞当前线程。

3.dispatch_sync触发死锁的条件

dispatch_sync触发死锁的条件:
当前队列必须是串行队列,且block任务追加进的目标任务队列必须跟当前队列是同一个串行队列。

相关文章

  • iOS -dispatch_sync阻塞的是线程还是队列?

    1.几个概念 1.1 进程--独立运转的公司 在iOS 中 一个进程就是一个正在运行的一个应用程序; 比如 QQ...

  • 多线程开发中的一些基本概念的理解

    进程、线程、队列(串行队列、并行队列)、同步(dispatch_sync)、异步(dispatch_async)、...

  • iOS_GCD的零零总总

    1、主线程队列 VS 分线程队列 dispatch_sync 和 dispatch_async 区别: dispa...

  • iOS dispatch几种死锁的验证

    示例一: 会发生死锁,简单判断就是外部线程是主线程,dispatch_sync到的是主队列,主队列是个串行队列。 ...

  • Swift GCD 的串行队列与并行队列

    队列异步是否阻塞当前线程同步是否阻塞当前线程执行顺序串行队列否是按添加顺序并行队列否是同时执行,但会被同步阻塞 串...

  • 阻塞队列和线程池

    1.阻塞队列 1)支持阻塞的插入方法:意思是当队列满时,队列会阻塞插入元素的线程,直到队列不满。2)支持阻塞的移除...

  • Day 38 并发线程池底层原理详解与源码分析

    任务先放到核心线程池,放不下了再放到阻塞队列,还是放不下就放到最大线程池中. 重点关注,阻塞队列,回收时间实现,拒...

  • 线程池

    [TOC] 线程池 1. 并发队列:阻塞队列和非阻塞队列 区别如下: 入队: 非阻塞队列:当队列中满了的时候,放入...

  • Java多线程之阻塞队列

    一基本概念:1:什么叫阻塞队列阻塞队列都是相对于非阻塞队列而言的,非阻塞队列就是队列不会对当前线程产生阻塞;例如当...

  • c++ 实现 blocking queue

    阻塞队列就是多线程线程安全的队列,在多线程场景下经常用到,c++ 标准库里面没有提供阻塞队列,boost 中有提供...

网友评论

      本文标题:iOS -dispatch_sync阻塞的是线程还是队列?

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