IOS-多线程技术
无论使用哪种多线程技术都可以使用
[NSThread currentThread]跟踪查看当前执行所在的线程情况。
num = 1表示在主线程上执行的任务
http://www.cnblogs.com/mcj-coding/p/3556419.html
================================================================
1. NSObject多线程技术
1> 使用performSelectorInBackground可以开启后台线程,执行selector选择器选择的方法
2> 使用performSelectorOnMainThread可以重新回到主线程执行任务,通常用于后台线程更新界面UI时使用
3> [NSThread sleepForTimeInterval:1.0f];
让当前线程休眠,通常在程序开发中,用于模拟耗时操作,以便跟踪不同的并发执行情况!
但是:在程序发布时,千万不要保留此方法!不要把测试中的代码交给客户,否则会造成不好的用户体验。
提示:使用performSelectorInBackground也可以直接修改UI,但是强烈不建议使用。
注意:在使用NSThread或者NSObject的线程方法时,一定要使用自动释放池,否则容易出现内存泄露。
================================================================
2. NSThread的多线程技术,仅了解即可,使用极少!
1> 类方法直接开启后台线程,并执行选择器方法
detachNewThreadSelector
2> 成员方法,在实例化线程对象之后,需要使用start执行选择器方法
initWithTarget
对于NSThread的简单使用,可以用NSObject的performSelectorInBackground替代
同时,在NSThread调用的方法中,同样要使用autoreleasepool进行内存管理,否则容易出现内存泄露。
================================================================
3. NSOperation,面向对象的多线程技术
1> 使用步骤:
1) 实例化操作
a) NSInvocationOperation
b) NSBlockOperation
2) 将操作添加到队列NSOperationQueue即可启动多线程执行
2> 更新UI使用主线程队列
[NSOpeationQueue mainQueue] addOperation ^{};
3> 操作队列的setMaxConcurrentOperationCount
可以设置同时并发的线程数量!
提示:此功能仅有NSOperation有!
4> 使用addDependency可以设置任务的执行先后顺序,同时可以跨操作队列指定依赖关系
提示:在指定依赖关系时,注意不要循环依赖,否则不工作。
================================================================
4. GCD,C语言
GCD就是为了在“多核”上使用多线程技术
1> 要使用GCD,所有的方法都是dispatch开头的
2> 名词解释
global 全局
queue 队列
async 异步
sync 同步
3> 要执行异步的任务,就在全局队列中执行即可
dispatch_async 异步执行控制不住先后顺序
4> 关于GCD的队列
全局队列 dispatch_get_global_queue
参数:优先级 DISPATCH_QUEUE_PRIORITY_DEFAULT
始终是 0
串行队列 dispatch_queue_create("myQueue", DISPATCH_QUEUE_SERIAL);
是创建得到的,不能直接获取
主队列 dispatch_get_main_queue
5> 异步和同步与方法名无关,与运行所在的队列有关!
提示:要熟悉队列于同步、异步的运行节奏,一定需要自己编写代码测试!
同步主要用来控制方法的被调用的顺序
实例代码
- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"%@", [NSThread currentThread]);
// 实例化操作队列
_queue = [[NSOperationQueue alloc] init];
}
#pragma mark - 操作
// 耗时操作演示
- (void)bigDemo
{
// 自动释放池
// 负责其他线程上的内存管理,在使用NSThread或者NSObject的线程方法时,一定要使用自动释放池
// 否则容易出现内存泄露。
@autoreleasepool {
// // 模拟网络下载延时
// for (NSInteger i = 0; i < 1000; i++) {
// NSString *str = [NSString stringWithFormat:@"%d", i];
//
// // 提示:NSLog是非常耗时的操作!
// NSLog(@"大任务-> %@", str);
// }
NSLog(@"%@", [NSThread currentThread]);
// 模拟网络下载延时,睡眠1秒,通常是在开发中测试使用。
[NSThread sleepForTimeInterval:1.0f];
// 强烈不建议直接在后台线程更新界面UI!
// 模拟获取到下载的图像
UIImage *image = [UIImage imageNamed:@"头像1"];
// 在主线程更新图像
// 使用self调用updateImage方法在主线程更新图像
// [self performSelectorOnMainThread:@selector(updateImage:) withObject:image waitUntilDone:YES];
// 使用imageView的setImage方法在主线程更新图像
[_imageView performSelectorOnMainThread:@selector(setImage:) withObject:image waitUntilDone:YES];
// ....
}
}
#pragma mark - 更新图像,模拟从网络上下载完图片后,更新界面的操作
- (void)updateImage:(UIImage *)image
{
NSLog(@"更新图像-> %@", [NSThread currentThread]);
_imageView.image = image;
}
#pragma mark - Actions
- (IBAction)bigTask
{
// 本方法中的所有代码都是在主线程中执行的
// NSObject多线程技术
NSLog(@"执行前->%@", [NSThread currentThread]);
// performSelectorInBackground是将bigDemo的任务放在后台线程中执行
[self performSelectorInBackground:@selector(bigDemo) withObject:nil];
NSLog(@"执行后->%@", [NSThread currentThread]);
// [self bigDemo];
NSLog(@"执行完毕");
}
- (IBAction)smallTask
{
NSString *str = nil;
for (NSInteger i = 0; i < 50000; i++) {
str = [NSString stringWithFormat:@"%d", i];
}
NSLog(@"小任务-> %@", str);
}
#pragma mark NSThread演练
- (IBAction)threadDemo
{
// 新建一个线程,调用@selector方法
// [NSThread detachNewThreadSelector:@selector(bigDemo) toTarget:self withObject:nil];
// 成员方法
NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(bigDemo) object:nil];
// 启动start线程
[thread start];
}
#pragma mark - NSOperation演练
- (void)opAction
{
NSLog(@"%@", [NSThread currentThread]);
// 模拟延时
[NSThread sleepForTimeInterval:1.0f];
// 模拟获取到图像
UIImage *image = [UIImage imageNamed:@"头像1"];
// 设置图像,在主线程队列中设置
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
_imageView.image = image;
}];
}
#pragma mark invocation
- (IBAction)operationDemo1
{
NSInvocationOperation *op1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(opAction) object:nil];
// 如果使用start,会在当前线程启动操作
// [op1 start];
// 1. 一旦将操作添加到操作队列,操作就会启动
[_queue addOperation:op1];
}
#pragma mark blockOperation
- (IBAction)operationDemo2
{
// 用block的最大好处,可以将一组相关的操作,顺序写在一起,便于调试以及代码编写
[_queue addOperationWithBlock:^{
NSLog(@"%@", [NSThread currentThread]);
// 模拟延时
[NSThread sleepForTimeInterval:1.0f];
// 模拟获取到图像
UIImage *image = [UIImage imageNamed:@"头像1"];
// 设置图像,在主线程队列中设置
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
_imageView.image = image;
}];
}];
}
#pragma mark 模仿下载网络图像
- (IBAction)operationDemo3:(id)sender
{
// 1. 下载
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"下载 %@" , [NSThread currentThread]);
}];
// 2. 滤镜
NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"滤镜 %@" , [NSThread currentThread]);
}];
// 3. 显示
NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"更新UI %@" , [NSThread currentThread]);
}];
// 添加操作之间的依赖关系,所谓“依赖”关系,就是等待前一个任务完成后,后一个任务才能启动
// 依赖关系可以跨线程队列实现
// 提示:在指定依赖关系时,注意不要循环依赖,否则不工作。
[op2 addDependency:op1];
[op3 addDependency:op2];
// [op1 addDependency:op3];
[_queue addOperation:op1];
[_queue addOperation:op2];
[[NSOperationQueue mainQueue] addOperation:op3];
}
#pragma mark 限制线程数量
- (IBAction)operationDemo4
{
// 控制同时最大并发的线程数量
[_queue setMaxConcurrentOperationCount:2];
for (NSInteger i = 0; i < 200; i++) {
NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"%@", [NSThread currentThread]);
}];
[_queue addOperation:op];
}
}
#pragma mark - GCD演练
- (IBAction)gcdDemo1
{
// 1. 队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
// 2. 将任务异步(并发)执行
dispatch_async(queue, ^{
NSLog(@"a->%@", [NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"b->%@", [NSThread currentThread]);
});
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"main - > %@", [NSThread currentThread]);
});
}
#pragma mark 串行队列
- (IBAction)gcdDemo2
{
// 1. 队列,串行队列需要自行创建,不能get
dispatch_queue_t queue = dispatch_queue_create("myQueue", DISPATCH_QUEUE_SERIAL);
dispatch_sync(queue, ^{
NSLog(@"a->%@", [NSThread currentThread]);
});
dispatch_sync(queue, ^{
NSLog(@"b->%@", [NSThread currentThread]);
});
}
@end
IOS-多线程技术的更多相关文章
- iOS多线程技术方案
iOS多线程技术方案 目录 一.多线程简介 1.多线程的由来 2.耗时操作的模拟试验 3.进程和线程 4.多线程的概念及原理 5.多线程的优缺点和一个Tip 6.主线程 7.技术方案 二.Pthrea ...
- iOS多线程技术
iOS多线程技术主要分配NSThread.NSOperation和GCD.下边来简单的介绍一下吧. 随性一点,就不按照顺序来了.所以先介绍一下NSOperation. ---------------- ...
- iOS 多线程技术2
iOS 多线程技术2 NSOperation NSInvocationOperation //创建一个队列 NSOperationQueue *queue = [[NSOperationQueue a ...
- 浅谈iOS多线程
浅谈iOS多线程 首先,先看看进程和线程的概念. 图1.1 这一块不难理解,重点点下他们的几个重要区别: 1,地址空间和资源:进程可以申请和拥有系统资源,线程不行.资源进程间相互独立,同一进程的各线程 ...
- iOS开发之多线程技术
本篇争取一篇讲清讲透,依然将通过四大方面清晰的对iOS开发中多线程的用法进行详尽的讲解: 一.什么是多线程 1)多线程执行原理 2)线程与进程 3)多线程的优缺点 二.我们为什么要用多线程编程技术 三 ...
- iOS的三种多线程技术NSThread/NSOperation/GCD
1.iOS的三种多线程技术 1.NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程) 2.以下两点是苹果专门开发的"并发"技术,使得程序员可以不再去关心 ...
- 多线程技术在iOS开发中的使用
进程和线程 要使用多线程,首先需要理解进程和线程这2个概念.这里我简单的说一下. 所谓进程对应的是一个应用程序,负责开辟内存空间供应用程序使用,但是进程不能执行任务(指令).一个进程至少包含一条线程, ...
- OS X 和iOS 中的多线程技术(上)
OS X 和iOS 中的多线程技术(上) 本文梳理了OS X 和iOS 系统中提供的多线程技术.并且对这些技术的使用给出了一些实用的建议. 多线程的目的:通过并发执行提高 CPU 的使用效率,进而提供 ...
- OS X 和iOS 中的多线程技术(下)
OS X 和iOS 中的多线程技术(下) 上篇文章中介绍了 pthread 和 NSThread 两种多线程的方式,本文将继续介绍 GCD 和 NSOperation 这两种方式.. 1.GCD 1. ...
- iOS的多线程技术
iOS的三种多线程技术 1.NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程) 2.以下两点是苹果专门开发的“并发”技术,使得程序员可以不再去关心线程的具体使用问题 ØN ...
随机推荐
- VC++6.0MFC运行的简单流程
背景: 由于下位机做的一些东西,总是需要通过上位机来验证,以及为了直观给客户展示下位机的功能,所以时常需要编写一些简单的APP.今天就以VC++6.0为例,简单的记录下该如何快速的创建一个APP. 正 ...
- 【PHP面向对象(OOP)编程入门教程】5.如何实例化对象?
我们上面说过面向对象程序的单位就是对象,但对象又是通过类的实例化出来的,既然我们类会声明了,下一步就是实例化对象了. 当定义好类后,我们使用new关键字来生成一个对象. $对象名称 = new 类名称 ...
- U盘快速装ghost系统
U盘装系统U盘装系统是目前最常用的系统安装方式.特别适合于未安装光驱的台式机或超薄笔记本电脑上.小编为了给这类无光驱用户提供最大的便利,将在本文中为大家讲解最详细的U盘装系统教程. 您需要准备一个空的 ...
- Java多线程文件下载
一. 多线程下载文件考虑处理步骤: 1. 如何获取文件的长度 2. 合理的创建线程数量,并计算每一个线程下载的长度 3. 如何将多个线程下载的字节写入到文件中 二. 代码实现如下: package c ...
- Java多线程基础知识(三)
一. 管道输入/输出流 它和其它文件输入/输出流或网络输入/输出流的不同之处,它主要是线程之间的数据传输,而传输的媒介是内存. 管道输入/输出流主要包含四中实现: 1. PipedOutputStre ...
- leetcode 32. Longest Valid Parentheses
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
- [KOJ6024]合并果子·改(强化版)
[COJ6024]合并果子·改(强化版) 试题描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多把这些果子堆排成一排,然后所有的果子合成一堆. 每一次合并 ...
- PingUtil in Android
Ping a host in Android:“ping -c 1 127.0.0.1”-c 1: The ping times. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 1 ...
- 线段树基础模板&&扫描线
线段树的单点更新+区间求和 hdu1166敌兵布阵 Input 第一行一个整数T,表示有T组数据. 每组数据第一行一个正整数N(N<=),表示敌人有N个工兵营地 ,接下来有N个正整数,第i个正整 ...
- 线程池大小 & cpu core
http://stackoverflow.com/questions/14556037/number-of-processor-core-vs-the-size-of-a-thread-pool ht ...