一、前言部分
最近在面试,重新温习了一遍多线程,希望加深一遍对于多线程的理解。
1、什么是进程?
1).要了解线程我们必须先了解进程,通俗来讲进程就是在系统中运行的一个应用程序。
2).每个线程之间是独立存在的,分别运行在其专用的且受保护的内存空间中。
3).比如打开QQ或Xcode系统会分别开启两个进程 如图:
4)、我们可以通过"活动监视器"查看Mac系统中所开启的进程。
2、什么是线程?
1).一个进程要想执行任务必须得有线程,即一个进程至少要有一个线程。
2).线程是进程的基本执行单元,一个进程(程序)的所有任务都是在线程中执行的。
3).比如使用酷狗播放音乐、使用迅雷下载电影都需要在线程中运行 如图:
3、什么是线程的串行?
1).一个线程中任务是串行执行的(顺序执行)的,也就是说一个线程同一时间内只能执行一个任务。
2).串行执行图解,比如一个线程下载3个文件(文件A、B、C)
4、什么是多线程?
1).一个进程中可以开启多个线程,每个线程可以并发(同时)执行不同的任务。
2).类似关系列举:进程---->车间;线程---->车间工人
3).多线程图解,比如同时开启3个线程分别下载3个文件(文件A、B、C)
5、多线程原理
1).同一时间CPU只能执行一个线程,只有一个线程在工作(执行)。
2).多线程并发(同时)执行,其实是CPU快速的在多个线程之间调度(切换)。
3).如果CPU调度线程的速度够快,就会造成多线程并发执行的假象。
4).多线程的缺点:
1、每个线程都会占用一定的内存空间(默认情况下:主线程占用1MB,子线程占用512KB),
如果开启线程过多会占用大量的内存空间因而造成程序性能降低。
2、线程越多CPU调度线程上的开销就越大(类似工厂工人越多,工厂开销也越大)。
3、使程序设计更复杂:比如多线程的数据通信,多线程之间的数据共享。
5).多线程的优点:
1、能适当提高程序的执行效率。
2、能适当提高资源利用率(CPU、内存的利用率)
6、什么是主线程?
1).一个iOS程序开启后默认会开启一个线程,这个线程被称为"主线程"或"UI线程"。
2).主线程的主要作用:
1、显示/刷新UI界面
2、处理UI事件(比如点击事件、滚动事件、拖拽事件等)
3).主线程注意点:
1、别将耗时的操作放在主线程中,耗时操作放在主线程中会造成程序卡顿的问题。
7、耗时操作Demo演示
1)、直接在主线程中运行的Demo
1 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
2 //获取当前执行方法和当前线程
3 //number==1 主线程
4 //number!=1 其他线程、子线程、次线程
5 NSLog(@"%s----%@",__func__,[NSThread currentThread]);
6
7 //直接在主线程中运行 造成UI操作卡顿
8 [self longTimeOperation];
9 }
10
11 #pragma mark-耗时操作
12 -(void)longTimeOperation{
13 for (int i=0; i<20000; i++) {
14 NSLog(@"%d",i);
15 }
16 }
2)、在子线程中运行的Demo
1 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
2 //获取当前执行方法和当前线程
3 //number==1 主线程
4 //number!=1 其他线程、子线程、次线程
5 NSLog(@"%s----%@",__func__,[NSThread currentThread]);
6
7 //将耗时操作放在子线程中执行,不影响UI的操作
8 [self performSelectorInBackground:@selector(longTimeOperation) withObject:nil];
9 }
10
11 #pragma mark-耗时操作
12 -(void)longTimeOperation{
13 for (int i=0; i<20000; i++) {
14 NSLog(@"%d",i);
15 }
16 }
8、iOS中多线程的实现方案
技术方案 | 简介 | 语言 | 线程生命周期 | 使用频率 |
pthread |
|
C | 程序员管理 | 几乎不用 |
NSThread |
|
OC | 程序员管理 | 偶尔使用 |
GCD |
|
C | 自动管理 | 经常使用 |
NSOperation |
|
OC | 自动管理 | 经常使用 |
9、pthread 的使用
1 #import <pthread.h>
2 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
3 [self test];
4 }
5
6 //使用pthread创建线程
7 -(void)test{
8 //声明一个线程变量
9 pthread_t threadID;
10
11 /*
12 参数:
13 1、要开的线程的变量
14 2、线程的属性
15 3、要在这个子线程中执行的函数(任务)
16 4、这个函数(任务)需要传递的参数
17 */
18 //pthread_create(pthread_t *restrict, const pthread_attr_t *restrict, void *(*)(void *), void *restrict);
19
20 id str=@"hello";
21 //id 需要转成void * 需要使用__bridge进行桥连
22 //1、这里只是临时把str对象转成void *在这里临时使用,不改变这个对象的所有权
23 //2、把对象所有权交出去,在这个函数把str转成void *
24 //如果使用MRC 这里不需要使用桥连可以直接使用这个参数
25 //ARC自动内存管理,本质是编译器特性,是在程序编译的时候,编译器在适合的地方帮我们添加retain、release、autorelease
26 pthread_create(&threadID, NULL, run, (__bridge void*)str);
27 }
28
29 #pragma mark-耗时操作
30 void *run(void *prama){
31 //void * 相当于OC里面的id
32 //__bridge 桥连操作
33 NSString *str=(__bridge NSString*)prama;
34 for (int i=0; i<20000; i++) {
35 NSLog(@"%d----%@----%@",i,[NSThread currentThread],str);
36
37 }
38 return NULL;
39 }
10、NSThread 的创建方式
1 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
2 //[self test1];
3
4 //[self test2];
5
6 [self test3];
7 }
8
9 #pragma mark-线程的创建方式
10 //创建线程方式1
11 -(void)test1{
12 //实例化一个线程对象
13 NSThread *thread=[[NSThread alloc]initWithTarget:self selector:@selector(run1) object:nil];
14 //线程启动方法
15 [thread start];
16 }
17
18 //创建线程方式2
19 -(void)test2{
20 //创建线程后自动启动线程
21 [NSThread detachNewThreadSelector:@selector(run2:) toTarget:self withObject:@"KK"];
22 }
23
24 //创建线程方式3
25 -(void)test3{
26 //隐式创建线程并启动
27 [self performSelectorInBackground:@selector(run2:) withObject:@"Hi"];
28
29 /*
30 test2、test3 两个线程创建方法的优缺点:
31 优点:简单快捷
32 缺点:无法对线程更详细的设置
33 */
34 }
35
36 #pragma mark-耗时操作
37 -(void) run1{
38 for (int i=0; i<200; i++) {
39 NSLog(@"%d----%@",i,[NSThread currentThread]);
40 }
41 }
42 -(void) run2:(NSString*)param{
43 for (int i=0; i<200; i++) {
44 NSLog(@"%d----%@---%@",i,[NSThread currentThread],param);
45 }
46 }
11、线程的属性介绍
1 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
2 [self test];
3 }
4
5 #pragma mark-线程的属性
6 -(void)test{
7 //实例化一个线程对象
8 NSThread *threadA=[[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
9 //线程名称,用于区分多个线程
10 threadA.name=@"Thread A";
11 //线程的优先级 0.0~1.0 默认值 0.5
12 //实际开发中一般不修改优先级的值
13 threadA.threadPriority=0.1;
14 //线程启动方法
15 [threadA start];
16
17 // //实例化一个线程对象
18 // NSThread *threadB=[[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
19 // //线程名称,用于区分多个线程
20 // threadB.name=@"Thread B";
21 // //线程的优先级 0.0~1.0 默认值 0.5
22 // threadB.threadPriority=1.0;
23 // //线程启动方法
24 // [threadB start];
25 }
26
27
28 #pragma mark-耗时操作
29 -(void) run{
30 for (int i=0; i<200; i++) {
31 NSLog(@"%d----%@",i,[NSThread currentThread]);
32 }
33 }
12、GCD的使用实例
1 #pragma mark - GCD演练
2 /**
3 并发队列,同步执行
4 */
5 - (void)gcdDemo4 {
6 // 1. 队列
7 dispatch_queue_t queue = dispatch_queue_create("itcast", DISPATCH_QUEUE_CONCURRENT);
8
9 // 2. 同步执行任务
10 for (int i = 0; i < 10; i++) {
11 dispatch_sync(queue, ^{
12 NSLog(@"%@ %d", [NSThread currentThread], i);
13 });
14 }
15 }
16
17 /**
18 并发队列,异步执行
19 */
20 - (void)gcdDemo3 {
21 // 1. 队列
22 dispatch_queue_t queue = dispatch_queue_create("itcast", DISPATCH_QUEUE_CONCURRENT);
23
24 // 2. 异步执行任务
25 for (int i = 0; i < 10; i++) {
26 dispatch_async(queue, ^{
27 NSLog(@"%@ %d", [NSThread currentThread], i);
28 });
29 }
30
31 }
32
33 /**
34 串行队列,异步执行
35 */
36 - (void)gcdDemo2 {
37 // 1. 队列
38 dispatch_queue_t queue = dispatch_queue_create("itcast", NULL);
39
40 // 2. 异步执行任务
41 for (int i = 0; i < 10; i++) {
42 dispatch_async(queue, ^{
43 NSLog(@"%@ %d", [NSThread currentThread], i);
44 });
45 }
46 }
47
48 /**
49 串行队列,同步执行(开发中非常少用)
50 */
51 - (void)gcdDemo1 {
52
53 // 1. 队列
54 // dispatch_queue_t queue = dispatch_queue_create("icast", DISPATCH_QUEUE_SERIAL);
55 dispatch_queue_t queue = dispatch_queue_create("icast", NULL);
56 NSLog(@"执行前----");
57
58 // 执行任务
59 for (int i = 0; i < 10; i++) {
60 NSLog(@"调度----");
61
62 // 在队列中"同步"执行任务,串行对列添加同步执行任务,会立即被执行
63 dispatch_sync(queue, ^{
64 NSLog(@"%@ %d", [NSThread currentThread], i);
65 });
66 }
67 NSLog(@"for 后面");
68 }
View Code
13、NSOperation的使用实例
1 #pragma mark - 基本演练
2 // MARK: - 线程间通讯
3 - (void)opDemo5 {
4 NSOperationQueue *q = [[NSOperationQueue alloc] init];
5 [q addOperationWithBlock:^{
6 NSLog(@"耗时操作 %@", [NSThread currentThread]);
7
8 [[NSOperationQueue mainQueue] addOperationWithBlock:^{
9 NSLog(@"更新UI %@", [NSThread currentThread]);
10 }];
11 }];
12 }
13
14 // MARK: - 更简单的
15 - (void)opDemo4 {
16 NSOperationQueue *q = [[NSOperationQueue alloc] init];
17
18 for (int i = 0; i < 10; i++) {
19 [q addOperationWithBlock:^{
20 NSLog(@"down %@ %@", [NSThread currentThread], @(i));
21 }];
22 }
23 }
24
25 // MARK: - NSBlockOperation
26 - (void)opDemo3 {
27 NSOperationQueue *q = [[NSOperationQueue alloc] init];
28
29 for (int i = 0; i < 10; i++) {
30 NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
31 NSLog(@"down %@ %@", [NSThread currentThread], @(i));
32 }];
33
34 [q addOperation:op];
35 }
36 }
37
38 // MARK: - NSInvocationOperation
39 - (void)opDemo2 {
40 NSOperationQueue *q = [[NSOperationQueue alloc] init];
41
42 for (int i = 0; i < 10; i++) {
43 NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downloadImage:) object:@(i)];
44
45 [q addOperation:op];
46 }
47 }
48
49 - (void)opDemo1 {
50 NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downloadImage:) object:@"Invocation"];
51
52 // start 会立即在当前线程执行 selector 方法
53 // [op start];
54
55 // 将操作添加到队列,会自动异步执行
56 NSOperationQueue *q = [[NSOperationQueue alloc] init];
57 [q addOperation:op];
58 }
59
60 - (void)downloadImage:(id)obj {
61 NSLog(@"%@ %@", [NSThread currentThread], obj);
62 }
14、线程的状态介绍
1 -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
2 [self test];
3 }
4
5 #pragma mark-线程的属性
6 -(void)test{
7 //实例化一个线程对象
8 NSThread *thread=[[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
9 //放到可调度线程池,等待被调度 这时候是准备就绪状态
10 [thread start];
11 }
12
13
14 #pragma mark-耗时操作
15 -(void) run{
16
17 NSLog(@"%s",__func__);
18 //线程进来就睡眠2秒
19 //[NSThread sleepForTimeInterval:2.0];
20
21 //睡到指定的时间点
22 [NSThread sleepUntilDate:[NSDate dateWithTimeIntervalSinceNow:5.0]];
23
24 for (int i=0; i<200; i++) {
25 //满足一个条件后,阻塞线程的执行
26 if (i==10) {
27 [NSThread sleepForTimeInterval:2.0];
28 }
29
30 //一旦达到某个条件,就强制终止线程的执行
31 if (i==100) {
32 //一旦终止就不能重新启动,后面的代码就不会被执行
33 [NSThread exit];
34 }
35 NSLog(@"%d----%@",i,[NSThread currentThread]);
36 }
37 }
15、多线程的安全隐患
1).一块资源可能会被多个线程共享,也就是多个线程可能访问同一块资源。
2).比如多个线程访问同一个对象、同一个变量、同一个文件。
3).当多个线程访问同一个资源时,很容易引发数据错乱和数据安全问题。
16、解决多线程安全问题
1).添加互斥锁解决多线程访问同一资源造成的数据安全问题。
2).互斥锁使用格式:@synchronized (self) {//需要锁定的代码}
3).互斥锁的优缺点:
优点:能有效防止因多线程抢夺同一资源造成的数据安全问题。
缺点:需要消耗大量的CPU资源(因此苹果不推荐使用互斥锁)
4).互斥锁的使用前提:多个线程抢夺同一资源。
17、利用属性的原子性保证数据的安全性
1).atomic 原子属性--默认属性
2).原子属性内部使用的是 自旋锁
3).自旋锁与互斥锁的异同
共同点:都可以锁定一段代码,同一时间内,只有一个线程能执行这段锁定的代码。
不同点:互斥锁 在锁定的时候其他线程在等待的时候会进入休眠 等待条件满足时需要重新唤醒。
自旋锁在锁定的时候,其他线程等待的时候做死循环一直等待条件满足,一旦条件满足就立马执行,
少了一个唤醒的过程。