最近遇到特别糟糕的面试,过程中提到多次对多线程的处理问题,并没有很好的给予答复和解决,所以在这里做个简单的备案:

期望能更加了解和熟练使用 多线程技术:

下面都是自己的总结,如果存在不对的,或者不足,请给予指正……

1. 多线程基本概念:方式 -> 同步 + 异步

1.1 同步 :  队列任务 在当前线程,按照添加先后顺序执行,不开辟新的线程;

1.1.1   同步方法 1

- (void)syncOne{

    NSLog(@"**************同步1 start ***************");
NSLog(@"同步1 start %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_SERIAL);
// 同步执行
dispatch_sync(queue, ^{
for (int i = 0; i < 3; i++) {
NSLog(@"同步1 %@",[NSThread currentThread]);
}
});
NSLog(@"同步1 end %@",[NSThread currentThread]);
NSLog(@"**************同步1 end ***************");
}

打印结果:::::

2017-12-19 16:56:06.537696+0800 DeadThread[6767:1309336] **************同步1 start ***************
2017-12-19 16:56:06.537841+0800 DeadThread[6767:1309336] 同步1 start <NSThread: 0x6040000789c0>{number = 1, name = main}
2017-12-19 16:56:06.537969+0800 DeadThread[6767:1309336] 同步1 <NSThread: 0x6040000789c0>{number = 1, name = main}
2017-12-19 16:56:06.538046+0800 DeadThread[6767:1309336] 同步1 <NSThread: 0x6040000789c0>{number = 1, name = main}
2017-12-19 16:56:06.538222+0800 DeadThread[6767:1309336] 同步1 <NSThread: 0x6040000789c0>{number = 1, name = main}
2017-12-19 16:56:06.538307+0800 DeadThread[6767:1309336] 同步1 end <NSThread: 0x6040000789c0>{number = 1, name = main}
2017-12-19 16:56:06.538404+0800 DeadThread[6767:1309336] **************同步1 end ***************

得到的结果::

1.创建了同步队列:dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_SERIAL);

1.线程在主线程执行

2.线程顺序执行的

1.1.2 同步方法 2

- (void)syncTwo{

    NSLog(@"**************同步2 start ***************");
NSLog(@"同步2 start %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_SERIAL);
// 同步执行
dispatch_async(dispatch_get_global_queue(0, 0), ^{//新开一个子线程 dispatch_sync(queue, ^{
for (int i = 0; i < 3; i++) {
NSLog(@"同步2 %@",[NSThread currentThread]);
}
}); }); NSLog(@"同步2 end %@",[NSThread currentThread]);
NSLog(@"**************同步2 end ***************");
}

打印结果::::

2017-12-20 10:08:49.982236+0800 DeadThread[8035:1857644] **************同步2 start ***************
2017-12-20 10:08:49.982373+0800 DeadThread[8035:1857644] 同步2 start <NSThread: 0x604000075480>{number = 1, name = main}
2017-12-20 10:08:49.982478+0800 DeadThread[8035:1857644] 同步2 end <NSThread: 0x604000075480>{number = 1, name = main}
2017-12-20 10:08:49.982512+0800 DeadThread[8035:1858341] 同步2 <NSThread: 0x60400026aa40>{number = 3, name = (null)}
2017-12-20 10:08:49.982563+0800 DeadThread[8035:1857644] **************同步2 end ***************
2017-12-20 10:08:49.982584+0800 DeadThread[8035:1858341] 同步2 <NSThread: 0x60400026aa40>{number = 3, name = (null)}
2017-12-20 10:08:49.982817+0800 DeadThread[8035:1858341] 同步2 <NSThread: 0x60400026aa40>{number = 3, name = (null)}

得到的结果:::::

1.线程在新开的线程执行;

2.线程内同步代码 运行是顺序进行的

1.1.3 同步死锁

解说一个经典的死锁代码:

- (void)viewDidLoad {
[super viewDidLoad]; NSLog(@"开始启动");// 1 dispatch_sync(dispatch_get_main_queue(), ^{ //2 NSLog(@"同步任务"); // 3
}); NSLog(@"启动结束"); //4
}

运行结果::::

2017-12-20 10:53:10.147278+0800 DeadThread[8403:2037075] 开始启动
(lldb)

解说原因::::

前提::1. viewDidLoad 任务是在主线程执行的  2. dispatch_sync(dispatch_get_main_queue(),^{ }); 在主线程 添加了新的任务

1.项目整体代码是 顺序运行的;

2.当代码 运行到 标记 1 时, 没有发生问题,log 正常启动,打印输出;

3.当代码 运行到 标记 2 时,添加了新的同步任务,且在同一个队列中(如图)

图示::

同步任务,view did load 任务 先添加的,所以先被执行,

运行到  标记 2 时,添加新的同步任务,此时 需要执行 这个同步任务,但此时  view did load 任务 还没有执行完毕;

造成的结果:

同步任务 需要等待 view did load 任务执行完毕;

view did load 任务 需要同步任务执行 完毕;

对不起,哥们,到了这里,程序死定了,呜呜……

1.2 异步 :具备开启新线程的能理,执行时,不影响主线程的运行

1.2.1  异步方法1

- (void)asyncOne{

    NSLog(@"**************异步1 start ***************");
NSLog(@"异步1 start %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_SERIAL);
// 异步执行
dispatch_async(queue, ^{
for (int i = 0; i < 3; i++) {
NSLog(@"异步1 %@",[NSThread currentThread]);
}
});
NSLog(@"异步1 end %@",[NSThread currentThread]);
NSLog(@"**************异步1 end ***************");
}

执行结果::::

2017-12-20 12:58:01.304660+0800 DeadThread[8552:2189332] **************异步1 start ***************
2017-12-20 12:58:01.304867+0800 DeadThread[8552:2189332] 异步1 start <NSThread: 0x600000078940>{number = 1, name = main}
2017-12-20 12:58:01.304959+0800 DeadThread[8552:2189332] 异步1 end <NSThread: 0x600000078940>{number = 1, name = main}
2017-12-20 12:58:01.304993+0800 DeadThread[8552:2189855] 异步1 <NSThread: 0x600000272140>{number = 3, name = (null)}
2017-12-20 12:58:01.305164+0800 DeadThread[8552:2189332] **************异步1 end ***************
2017-12-20 12:58:01.305184+0800 DeadThread[8552:2189855] 异步1 <NSThread: 0x600000272140>{number = 3, name = (null)}
2017-12-20 12:58:01.305255+0800 DeadThread[8552:2189855] 异步1 <NSThread: 0x600000272140>{number = 3, name = (null)}

得到的结果:::

1.不影响主线程的执行;

2.新任务的代码,在新的线程执行;

1.2.2 异步方法2

- (void)asyncTwo{

    NSLog(@"**************异步2 start ***************");
NSLog(@"异步2 start %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_queue_create("test", DISPATCH_QUEUE_CONCURRENT);
// 异步执行
dispatch_async(queue, ^{
for (int i = 0; i < 3; i++) {
NSLog(@"异步2 %@",[NSThread currentThread]);
}
});
NSLog(@"异步2 end %@",[NSThread currentThread]);
NSLog(@"**************异步2 end ***************");
}

执行结果:::

2017-12-20 13:16:36.866651+0800 DeadThread[8737:2276498] **************异步2 start ***************
2017-12-20 13:16:36.866783+0800 DeadThread[8737:2276498] 异步2 start <NSThread: 0x600000076540>{number = 1, name = main}
2017-12-20 13:16:36.866890+0800 DeadThread[8737:2276498] 异步2 end <NSThread: 0x600000076540>{number = 1, name = main}
2017-12-20 13:16:36.866891+0800 DeadThread[8737:2276887] 异步2 <NSThread: 0x608000079800>{number = 3, name = (null)}
2017-12-20 13:16:36.866960+0800 DeadThread[8737:2276498] **************异步2 end ***************
2017-12-20 13:16:36.866980+0800 DeadThread[8737:2276887] 异步2 <NSThread: 0x608000079800>{number = 3, name = (null)}
2017-12-20 13:16:36.867038+0800 DeadThread[8737:2276887] 异步2 <NSThread: 0x608000079800>{number = 3, name = (null)}

得到结果:::

1.不影响主线程的执行;

2.新任务的代码,在新的线程执行;

iOS 多线程的简单理解(1) 方式 :同步 异步的更多相关文章

  1. iOS 多线程的简单理解(3)执行方式 + 执行对列 的组合

    通过对前面两偏线程理解的总结,自己对线程的理解也逐渐加深,梳理的清晰起来…… 通常在使用线程 的时候,都是要用到 执行对列,执行方式,执行任务, 现在开始新一轮的深入 3. 1. 1  同步 + 串行 ...

  2. iOS 多线程的简单理解(4) 线程锁的简单使用

    要用到多线程 ,就不得不考虑,线程之间的交互,线程是否安全 推荐一个原文链接 是关于 线程锁的基本使用的  http://blog.csdn.net/qq_30513483/article/detai ...

  3. iOS 多线程的简单理解(2) 队列 :串行 ,并行,MainQueue,GlobalQueue

    多线程队列是装载线程任务的队形结构.(系统以先进先出的方式调度队列中的任务执行 FIFO).在GCD中有两种队列: 串行队列.并发队列. 队列 :串行队列.并发队列,全局主对列,全局并发队列 2.1. ...

  4. C#多线程的简单理解

    一.CLR线程池基础 创建和销毁线程是一个昂贵的操作,所以CLR管理了一个线程池(thread pool),可以将线程池看成一个黑盒. CLR初始化时,线程池中是没有线程的.线程的初始化与其他线程一样 ...

  5. iOS On-Demand Resources简单理解

    ios9引入了一个新功能,On-Demand Resources,它是app thinning 的一部分.这个机能简单的说,就是在下载app的时候,app中包含的不重要资源不下载,等到需要时,在由系统 ...

  6. ios -RunLoop(简单理解)

    一. RunLoop简介 RunLoop字面意思是运行时,即跑圈得意思.它可以在我们需要的时候自己跑起来运行,在我们没有操作的时候就停下来休息,充分节省CPU资源,提高程序性能. 二. RunLoop ...

  7. iOS NSRunloop的简单理解

    最近学习了下NSRunloop. 作一下简单的理解: 1.runloop与线程的关系,每一个线程创建是都会有伴有一个runloop诞生,runloop用来接收事件源,让线程执行事件.当没有事件处理时, ...

  8. iOS:GCD理解1(同步-异步、串行-并行)

    1.并行-异步(ST1与ST2抢占资源) 1-1).获取 并行(全局)队列 ,DISPATCH_QUEUE_PRIORITY_DEFAULT 为默认优先级. dispatch_queue_t queu ...

  9. ios多线程的几种创建方式以及基本使用

    1.第一种,NSThread方法,此方法使用很少! 2.NSOperation,面向对象的多线程技术 3. NSObject多线程技术 4.苹果推荐使用的GCD技术 5.GDC中的队列

随机推荐

  1. 洛谷 AT2434 JOI 公園 (JOI Park) 题解

    人生第一次AC黑题,我太感动了. 每日一题 day31 打卡 Analysis 先跑遍DJ,求出1到 i的最短路.得到每个点到 1号点的距离后,从小到大排序一遍,这时便可以枚举每个点到 1号点的距离修 ...

  2. Oracle NVL 函数 nvl nvl2

    Oracle中函数以前介绍的字符串处理,日期函数,数学函数,以及转换函数等等,还有一类函数是通用函数.主要有:NVL,NVL2,NULLIF,COALESCE,这几个函数用在各个类型上都可以. 下面简 ...

  3. PHP文件载入

    一.介绍 在实际开发中,经常会在一个PHP文件中引入其他的文件,被引入的文件可以是HTML文档,也可以是PHP文件 二.语法 require(文件名)和require_once(文件名) includ ...

  4. Pytest权威教程22-优质集成实践

    目录 优质集成实践 使用pip安装包 Python测试发现的约定 选择测试布局结构/导入规则 在应用程序代码外测试 测试作为应用程序代码的一部分 tox 返回: Pytest权威教程 优质集成实践 使 ...

  5. phpstrom php出现404

    phpstrom搭配xampp用着很舒服,但是配置不小心容易出现404错误,解决并不难,几步就完成了 !第一步:依次选择Tools-Deployment-configuration进入后如果为空,点击 ...

  6. 深度学习面试题18:网中网结构(Network in Network)

    目录 举例 参考资料 网中网结构通过多个分支的运算(卷积或池化),将分支上的运算结果在深度上连接 举例 一个3*3*2的张量, 与3个1*1*2的卷积核分别same卷积,步长=1, 与2个2*2*2的 ...

  7. 如何提高工具开发和数据分析的效率?| jupyter | Rstudio server

    这部分是超级干货,也能直接体现一个开发分析者的能力. 主要分为两部分: 1. 面对新问题时,如何高效的分析和开发? 2. 面对相似的问题时,如何最快时间的利用之前的开发经验? 因为现在我主要用shel ...

  8. java 判断list是否为空

    问题: 之前用 list!=null 来判断list是否为空,但发现,定义一个list后,即使里面并没有加入任何元素,返回的结果仍旧是 true, 其实,本意是希望在没有任何元素时,返回 false, ...

  9. 笔记 - DBSherlock: A Performance Diagnostic Tool for Transactional Databases

    OLTP系统的问题很难排查和定位,这就是为什么要花那么多钱去请DBA 因为TP系统的请求很多都是毫秒级别,而且同时有大量的并发,所以由于资源,或随机的原因导致的问题,很难去定位根因 哪怕数据库系统尤其 ...

  10. cv2.imread()

    cv2.imread() 使用opencv和caffe的伙伴们,可能会有一个疑问,那就是对于同时读取图片的cv2.imread()和caffe.io.loadimage两个函数,有什么差别? 1.cv ...