iOS中多线程的实现方案
什么是主线程?
一个iOS程序运行后,默认会开启一条线程,称为“主线程”或“UI线程”
主线程的主要作用
1.显示/刷新UI界面
2.处理UI事件(比如点击事件,滚动事件,拖拽事件)
主线程的使用注意
1.别将比较耗时的操作放在主线程中
2.耗时操作会卡在主线程中,严重影响UI的流畅程度

如图,将耗时操作放在主线程中,任务会按照串行顺序执行,在第五秒点击按钮之后,界面会卡住5秒
因为耗时操作还没有执行完,不能立即响应按钮的点击

1.pthread的使用
void *run(void *parme) {
NSLog(@"%@",[NSThread currentThread]);
for (int i = ; i < ; i++) {
NSLog(@"%d",i);
}
return NULL;
}
- (IBAction)btnClick:(id)sender {
pthread_t thread;
pthread_create(&thread, NULL, run, NULL);
}
2.NSThread的使用
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
[self createThread3];
}
//第一种创建方法
- (void)createThread1 {
//需要几个线程就alloc几个
NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(run:) object:@"第一种"];
thread.name = @"one_thread";
[thread start];
}
//第二种创建方法
- (void)createThread2 {
[NSThread detachNewThreadSelector:@selector(run:) toTarget:self withObject:@"第二种"];
}
//第三种创建方法
- (void)createThread3 {
[self performSelectorInBackground:@selector(run:) withObject:@"第三种"];
}
- (void)run:(NSString *)param {
NSLog(@"______%@_____%@",param,[NSThread currentThread]);
}
3.GCD的使用
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
[self syncMain];
}
/**
同步函数+主队列
*/
- (void)syncMain {
dispatch_queue_t queue = dispatch_get_main_queue();
NSLog(@"syncMain ---- begin");
//将任务加入到队列
dispatch_sync(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_sync(queue, ^{
NSLog(@"2----%@",[NSThread currentThread]);
});
dispatch_sync(queue, ^{
NSLog(@"3----%@",[NSThread currentThread]);
});
NSLog(@"syncMain ---- end");
}
/**
异步函数+主队列
*/
- (void)asyncMain {
//异步函数用在主队列上就不会开线程了
//获得串行队列
dispatch_queue_t queue = dispatch_get_main_queue();
//将任务加入到队列
dispatch_async(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"2----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"3----%@",[NSThread currentThread]);
});
}
/**
同步函数+串行队列:不会开启新的线程,在当前线程执行任务
*/
- (void)syncSerial {
//创建串行队列
dispatch_queue_t queue = dispatch_queue_create("com.520.queue", DISPATCH_QUEUE_SERIAL);
//将任务加入到队列
dispatch_sync(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_sync(queue, ^{
NSLog(@"2----%@",[NSThread currentThread]);
});
dispatch_sync(queue, ^{
NSLog(@"3----%@",[NSThread currentThread]);
});
}
/**
异步函数+串行队列:会开启新的线程,但是任务是串行的,执行完一个任务,再执行下一个任务
*/
- (void)asyncSerial {
//创建串行队列
dispatch_queue_t queue = dispatch_queue_create("com.520.queue", DISPATCH_QUEUE_SERIAL);
//将任务加入到队列
dispatch_async(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"2----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"3----%@",[NSThread currentThread]);
});
}
/**
同步函数+并发队列:不会开启线程,不能
*/
- (void)syncConcurrent {
//获得全局的并发队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//将任务添加到队列
dispatch_async(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
dispatch_async(queue, ^{
NSLog(@"1----%@",[NSThread currentThread]);
});
}
/**
异步函数+并发队列:可以同时开启多条线程
*/
- (void)asycConcurrent {
//创建一个队列
//第一个参数是标签等同于名字
//第二个参数传串行还是并行
// dispatch_queue_t queue = dispatch_queue_create("img.download", DISPATCH_QUEUE_CONCURRENT);
//获得全局的并发队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//将任务添加到队列
dispatch_async(queue, ^{
for (int i = ; i < ; i++) {
NSLog(@"1----%@",[NSThread currentThread]);
}
});
//将任务添加到队列
dispatch_async(queue, ^{
for (int i = ; i < ; i++) {
NSLog(@"2----%@",[NSThread currentThread]);
}
});
//将任务添加到队列
dispatch_async(queue, ^{
for (int i = ; i < ; i++) {
NSLog(@"3----%@",[NSThread currentThread]);
}
});
}
在使用GCD时,主要分为两步
1.定制任务
2.将任务添加到队列
这里还需要区分下同步,异步,并行,串行
同步异步:影响是否开启新的线程
并行串行:影响任务的执行方式
4.NSOperation的使用
NSOperation是抽象类,并不具备封装操作,必须使用它的子类
使用NSOperation子类的方式有三种
1.NSInvocationOperation
2.NSBlockOperation
3.自定义子类继承NSOperation,实现内部响应的方法
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
//创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
//创建任务1(invocationOperation方法)
NSInvocationOperation *op1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download1) object:nil];
//创建任务2(blockOperation方法)
NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"download2----%@",[NSThread currentThread]);
}];
[op2 addExecutionBlock:^{
NSLog(@"download3----%@",[NSThread currentThread]);
}];
//创建任务3(自定义方法)
DDZOperation *op3 = [[DDZOperation alloc] init];
//添加任务到队列中
[queue addOperation:op1];//内部自动调start方法
[queue addOperation:op2];
[queue addOperation:op3];
}
- (void)download1 {
NSLog(@"download1----%@",[NSThread currentThread]);
}
补充在自定义的DDZOperation中只有实现main方法才会开启线程处理任务
@implementation DDZOperation
- (void)main {
NSLog(@"自定义-----%@",[NSThread currentThread]);
}
iOS中多线程的实现方案的更多相关文章
- iOS中多线程知识总结(一)
这一段开发中一直在处理iOS多线程的问题,但是感觉知识太散了,所以就把iOS中多线程的知识点总结了一下. 1.基本概念 1)什么是进程?进程的特性是什么? 进程是指在系统中正在运行的一个应用程序. ...
- iOS中多线程原理与runloop介绍
一.线程概述 有些程序是一条直线,起点到终点:有些程序是一个圆,不断循环,直到将它切断.直线的如简单的Hello World,运行打印完,它的生命周期便结束了,像昙花一现那样:圆如操作系统,一直运行直 ...
- iOS 中多线程的简单使用
iOS中常用的多线程操作有( NSThread, NSOperation GCD ) 为了能更直观的展现多线程操作在SB中做如下的界面布局: 当点击下载的时候从网络上下载图片: - (void)loa ...
- IOS中多线程的总结
首先要知道线程和进程的区别.一个系统上运行的每一个应用程序都是一个线程.而进程中要执行的任务都是在线程上来实现的,所以说线程是进程的最小执行单元. 进程最少要有一个线程.多线程,顾名思义就是多条线程. ...
- iOS中多线程常用的知识点
1.pThread 跨平台的多线程技术 , 是IEEE制定的POSIX 表示可移植性操作系统接口的多线程计数,UNIX内核平台 Unix,Linux,Mac(小红帽) (windows上有可移 ...
- C# 中 多线程同步退出方案 CancellationTokenSource
C# 中提供多线程同步退出机制,详参对象: CancellationTokenSource CancellationTokenSource 中暂未提供复位操作,因此当调用Cancle 之后,若再次调用 ...
- iOS中多线程知识总结(二)
1.GCD GCD全称是Grand Central Dispatch,译为"强大的中枢管理器" 1)什么是任务?什么是队列? 任务和队列是GCD的核心. 任务: 执行什么操作 队列 ...
- ios中多线程GCD NSOperation NSThread 相关的操作解析
//1.GCD 继承自C语言 优点 简单方便 //开启一个子线程处理耗时的操作 dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIO ...
- OS X 和iOS 中的多线程技术(上)
OS X 和iOS 中的多线程技术(上) 本文梳理了OS X 和iOS 系统中提供的多线程技术.并且对这些技术的使用给出了一些实用的建议. 多线程的目的:通过并发执行提高 CPU 的使用效率,进而提供 ...
随机推荐
- ASP.NET 5 和Entity Framework 7公告仓库
ASP.NET 5 有一个公告仓库来介绍ASP.NET 5和 Entity Framework 7的主要变更说明,这个对于日夜更新的项目来说,很多人经常会遇到问题但是不知道去哪里寻找帮助,很多同学在做 ...
- EQueue文件持久化消息关键点设计思路
要持久化的关键数据有三种 消息: 队列,队列中存放的是消息索引信息,即消息在文件中的物理位置(messageOffset)和在队列中的逻辑位置(queueOffset)的映射信息: 队列消费进度,表示 ...
- 写给.NET开发者的数据库Migration方案
微软给我们提供了一种非常好用的数据库迁移方案,但是我发现周围的同学用的并不多,所以我还是想把这个方案整理一下..NET选手看过来,特别是还在通过手工执行脚本来迁移数据库的同学们,当然你也可以选择EF的 ...
- Assertor用于判断参数和抛出异常
代码 /// <summary> 断言器,用于判断和抛出异常 /// </summary> static class Assertor { /// <summary> ...
- AngularJs之八
***今天讲一下angularJs的路由功能: 一:angularJs路由. 1.AngularJS 路由允许我们通过不同的 URL 访问不同的内容. 2.通过 AngularJS 可以实现多视图的单 ...
- Java发送Http请求并获取状态码
通过Java发送url请求,查看该url是否有效,这时我们可以通过获取状态码来判断. try { URL u = new URL("http://10.1.2.8:8080/fqz/page ...
- ES6 新特性
ECMAScript 6(以下简称ES6)是JavaScript语言的下一代标准.因为当前版本的ES6是在2015年发布的,所以又称ECMAScript 2015. 也就是说,ES6就是ES2015. ...
- WCF学习之旅—WCF服务部署到IIS7.5(九)
上接 WCF学习之旅—WCF寄宿前的准备(八) 四.WCF服务部署到IIS7.5 我们把WCF寄宿在IIS之上,在IIS中宿主一个服务的主要优点是在发生客户端请求时宿主进程会被自动启动,并且你可以 ...
- PHP 高级编程(5/5) - SPL 数组重载
ArrayAccess接口 ArrayAccess接口是对象的行为看起来像个数组,定义了四个方法.接口概要如下: ArrayAccess { /* Methods */ abstract public ...
- 基于MySQL MEB的备份恢复
MEB(MySQL Enterprise Backup)是MySQL商业版中提供的备份工具,属于物理备份. 同XtraBackup一样,mysqlbackup的使用过程同样包含如下三个步骤: 备份(- ...