NSThread

NSThread是一个苹果封装过的,面向对象的线程对象。但是它的生命周期需要我们自己来手动管理,所以使用不是很常见,比如[NSThread currentThread],它可以获取当前的线程类,你就可以知道当前线程的各种属性。

创建并启动

    //1.先创建,再启动
//创建
NSThread *thread = [[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
//启动
[thread start]; //2.创建并自动启动
[NSThread detachNewThreadSelector:@selector(run) toTarget:self withObject:nil];
//苹果认为performSelector系列方法不安全,所以在swift中,去掉了这个方法
[self performSelectorInBackground:@selector(run) withObject:nil];

NSThread其他常用的方法

    /*
//3.NSThread的常见方法
//取消线程
- (void)cancel;
//启动线程
- (void)start;
//判断某个线程的状态的属性
@property (readonly, getter=isExecuting) BOOL executing;
@property (readonly, getter=isFinished) BOOL finished;
@property (readonly, getter=isCancelled) BOOL cancelled;
//设置和获取线程名字
-(void)setName:(NSString *)n;
-(NSString *)name;
//获取当前线程信息
+ (NSThread *)currentThread;
//获取主线程信息
+ (NSThread *)mainThread;
//使当前线程暂停一段时间,或者暂停到某个时刻
+ (void)sleepForTimeInterval:(NSTimeInterval)time;
+ (void)sleepUntilDate:(NSDate *)date;
*/

GCD

Grand Central Dispatch,感觉很强大,当然,它的确很强大。它是苹果公司为多处理器的并行运算提出的解决方案,所以可以自动合理的利用跟多的cpu内核(不懂~~~),GCD可以自动的管理线程的生命周期(创建线程,调度任务,销毁线程),作为程序员,我们不需要管理线程的生命周期,我们只需要告诉GCD我们需要完成什么任务。GCD使用了很多的Block(闭包),所以使用起来也很方便,基本上GCD的使用是比较多的。

任务和队列

概念:

任务

  任务就是操作,你要完成什么任务,就是你要完成什么操作,也就是一段代码。这段代码可以写到GCD中的Block中,所以这样添加任务特别方便。

  任务有两种执行方式:同步执行和异步执行的主要区别是会不会阻塞当前线程,知道Block中的代码执行完毕。

如果是同步(sync)操作,它会阻塞当前线程并等待Block中的任务执行完毕,然后当前线程才能继续执行。

如果是异步(async)操作,当前线程会继续往下执行,它不会阻塞当前线程。

队列

  队列是用来存放任务的。队列分为两种,串行队列和并行队列。

  • 串行队列

  串行队列中的任务会根据队列的定义FIFO的执行,一个接一个的先进先出进行执行。放到串行队列中的任务,GCD会根据FIFO(先进先出)地取出来一个,执行一个,然后去下一个,执行下一个,这样一个接一个的执行。

  • 并行队列

  并行队列中的任务会根据同步和异步的不同有不同的执行方式。

并行队列 中的任务根据同步或异步有不同的执行方式。虽然很绕,但请看下表

在串行队列中执行同步任务:不会新建线程,按顺序执行任务(毫无用处)
在串行队列中执行异步任务,会新建线程,按顺序执行任务(非常有用)
在并行队列中执行同步任务:不会新建线程,按顺序执行任务(几乎没用)
在并行队列中执行异步任务:会新建多个线程,但是无法确定任务的执行顺序(有用,但是很容易出错)
 

创建队列和创建任务

 
    //GCD
//创建队列
//主队列,是一个特殊的串行队列,它用于刷新UI,如何需要刷新UI的工作都要在主队列中进行,所以一般耗时的任务都要放在别的线程中完成。
dispatch_queue_t queue = dispatch_get_main_queue(); //自己创建的队列:自己可以创建串行队列,也可以创建并行队列。其中第一个参数是标识符,用于DEBUG的时候标识唯一的队列,可以为空。第二个参数很重要,用来表示创建的队列是并行的还是串行的,传入DIS或者NULL表示创建了一个串行队列。传入DIS表示创建了一个并行队列。
dispatch_queue_t queue2 = dispatch_queue_create("dl.test.QueueTest2", DISPATCH_QUEUE_SERIAL); dispatch_queue_t queue3 = dispatch_queue_create("dl.test.QueueTest3", DISPATCH_QUEUE_CONCURRENT); //创建任务
//创建同步任务:不会另开线程(SYNC)
dispatch_sync(queue2, ^{
NSLog(@"%@",[NSThread currentThread]);
}); //异步任务:会另开线程(ASYNC)
dispatch_async(queue3, ^{
NSLog(@"%@",[NSThread currentThread]);
});

这里我们看两个例子,对于GCD中,同步任务的使用问题:

    NSLog(@"之前-%@", [NSThread currentThread]);

    dispatch_sync(dispatch_get_main_queue(), ^{
NSLog(@"sync - %@", [NSThread currentThread]);
}); NSLog(@"之后-%@", [NSThread currentThread]); /*
分析:
1.打印完第一句后,dispatch_sync 就会立即阻塞当前的主线程,然后把block中的任务放到main_queue中,可是main_queue中的任务会被取出来放到主线程中执行,但是,主线程现在已经被dispatch_sync阻塞了,所以block中的代码就不能完成,不能完成,dispatch_sync就会一直阻塞主线程,造成死锁现象,导致主线程卡死。
*/

第二个例子

dispatch_queue_t queue = dispatch_queue_create("myQueue", DISPATCH_QUEUE_SERIAL);
NSLog(@"之前-%@",[NSThread currentThread]);
dispatch_async(queue, ^{
NSLog(@"sync之前-%@",[NSThread currentThread]);
dispatch_sync(queue, ^{
NSLog(@"sync-%@",[NSThread currentThread]);
});
NSLog(@"sync之后-%@",[NSThread currentThread]);
});
NSLog(@"之后-%@",[NSThread currentThread]);
/*
分析:
1.首先创建一个叫做“myQueue”的串行队列。
2.打印当前线程
3.在myQueue的串行队列中添加异步任务(async),异步执行,不会阻塞当前线程,于是有了两条线程,一个线程继续打印“之后-%@”这句。另一个线程执行执行block中的内容,首先打印“sync之前-%@”
4.接下来,dispatch_sync是同步任务,它会阻塞当前所在线程,一直等到sync里的任务执行完成后才会继续往下。于是 sync 就高兴的把自己 Block 中的任务放到 queue 中,可谁想 queue 是一个串行队列,一次执行一个任务,所以 sync 的 Block 必须等到前一个任务执行完毕,可万万没想到的是 queue 正在执行的任务就是被 sync 阻塞了的那个。于是又发生了死锁。
*/

队列组

将很多的队列添加到一个组里,当组里的所有任务都执行完成后,队列组会调用方法通知我们。

    //1.创建队列组
dispatch_group_t group = dispatch_group_create();
//2.创建队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//3.多次使用队列组的方法执行任务,只有异步方法
//3.1 执行3次循环
dispatch_group_async(group, queue, ^{
for (NSInteger i = ; i < ; i++) {
NSLog(@"group-01 - %@", [NSThread currentThread]);
}
});
//3.2 主队列执行8次循环
dispatch_group_async(group, queue, ^{
for (NSInteger i = ; i < ; i++) {
NSLog(@"group-02 - %@", [NSThread currentThread]);
}
});
//3.3 执行5次循环
dispatch_group_async(group, queue, ^{
for (NSInteger i = ; i < ; i++) {
NSLog(@"group-03 - %@", [NSThread currentThread]);
}
});
//4.都执行完成后自动通知
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
NSLog(@"完成 - %@", [NSThread currentThread]);
});

回到主线程

在其他线程中操作完成后,必须回到主线程中更新UI,下边代码是回到主线程的方法:

GCD

//Objective-C
dispatch_async(dispatch_get_main_queue(), ^{
});

iOS中的多线程基础的更多相关文章

  1. OS X 和iOS 中的多线程技术(上)

    OS X 和iOS 中的多线程技术(上) 本文梳理了OS X 和iOS 系统中提供的多线程技术.并且对这些技术的使用给出了一些实用的建议. 多线程的目的:通过并发执行提高 CPU 的使用效率,进而提供 ...

  2. OS X 和iOS 中的多线程技术(下)

    OS X 和iOS 中的多线程技术(下) 上篇文章中介绍了 pthread 和 NSThread 两种多线程的方式,本文将继续介绍 GCD 和 NSOperation 这两种方式.. 1.GCD 1. ...

  3. iOS中数据库应用基础

    iOS 数据库入门 一.数据库简介 1.什么是数据库? 数据库(Database) 是按照数据结构来组织,存储和管理数据的仓库 数据库可以分为2大种类 关系型数据库(主流) PC端 Oracle My ...

  4. iOS中的多线程 NSOperation

    在ios中,使用多线程有三种方式,分别是:NSThread.NSOperation和NSOperationQueue.GCD,在本节,主要讲解一下NSOperation的使用. NSOperation ...

  5. IOS中的多线程和NSRunLoop概述(转载)

    线程概述 有些程序是一条直线,从起点到终点,如Hello World,运行打印完,它的生命周期便结束了:有些程序是一个圆,不断循环,直到将它切断,如操作系统,一直运行直到你关机.  一个运行着的程序就 ...

  6. java中的多线程 // 基础

    java 中的多线程 简介 进程 : 指正在运行的程序,并具有一定的独立能力,即 当硬盘中的程序进入到内存中运行时,就变成了一个进程 线程 : 是进程中的一个执行单元,负责当前程序的执行.线程就是CP ...

  7. C#中的多线程 - 基础知识 z

    原文:http://www.albahari.com/threading/ 专题:C#中的多线程 1简介及概念Permalink C# 支持通过多线程并行执行代码,线程有其独立的执行路径,能够与其它线 ...

  8. iOS中的多线程NSThread/GCD/NSOperation & NSOperationQueue

    iOS多线程有四套多线程方案: Pthreads NSThread GCD NSOperation & NSOperationQueue 接下来我来一个一个介绍他们 Pthreads 在类Un ...

  9. C#中的多线程 - 基础知识

    原文:http://www.albahari.com/threading/ 文章来源:http://blog.gkarch.com/threading/part1.html 1简介及概念 C# 支持通 ...

随机推荐

  1. CENTOS7小注意

    由于第一次使用 Linux CENTOS ,所以安装了图形界面,但是在终端执行yum 安装的时候,总是提示 Existing lock /var/run/yum.pid: another copy i ...

  2. [妙味JS基础]第九课:定时器管理、函数封装

    知识点总结 函数封装 回调函数 实例:抖动函数 获取当前的位置 通过数组来实现,一正一负,直到恢复成0为止. 当前位置与数组中各值相加

  3. 索引图像(X与map)的显示、保存、转化

    有的图像载入后,出现X.map两个矩阵,那么他就是索引图像. load wbarb; figure,imshow(X,map);%显示原图 imwrite(X,map,'C:\Users\Jv\Des ...

  4. 从零开始学Axure原型设计(进阶篇)

    Axure不仅能制作静态的视觉稿.页面,还能添加交互动作,是进行原型设计的最佳软件之一.在认识了Axure的界面和部件库之后,我们可以用它来画线框图了,但是静态的线框图在表达上不如有交互的原型图来得直 ...

  5. ffmpeg编译

    CFLAGS=-g ./configure --enable-opengl  --disable-yasm  --enable-shared --enable-pic

  6. dell 去鼠标版功能widnows

    桌面计算机(点击右键)----管理----设备管理器-----鼠标------选择触摸板(ps/2 兼容鼠标)---右击------跟新驱动-------浏览计算机查找------从计算机列表中选择- ...

  7. Objective-C中的instancetype与id的区别

    一.什么是instancetype instancetype是clang 3.5开始,clang提供的一个关键字,表示某个方法返回的未知类型的Objective-C对象.我们都知道未知类型的的对象可以 ...

  8. 27.编写一个Animal类,具有属性:种类;具有功能:吃、睡。定义其子类Fish 和Dog,定义主类E,在其main方法中分别创建其对象并测试对象的特性。

    ///Animal类 package d922A; public class Animal { private String kind; public String getKind() { Syste ...

  9. 【实验室笔记】C#上位机学习笔记

    用C#编写上位机,基本流程是[1]串口配置,[2]串口发送数据,[3]串口接收数据. [1]串口配置 串口的属性配置包括: No.1串口端口号 No.2串口波特率 No.3串口数据位 No.4串口停止 ...

  10. zstu 4215 多起点bfs

    input n m  1<=n,m<=1000 n*m的地图,全为大写字母 7 10 WWWWWCCDEW WWWWCCEEEW WTWWWCCCCW WWFFFFFFWW WWFAAAA ...