Object-C关于GCD多线程的使用

 

```objc
1 使用Crearte函数建立的并发队列和全局并发队列的主要区别:
1)全局并发队列在整个应用程序中自己是默认存在的而且对应有高优先级、默认优先级、低优先级和后台优先级一共四个并发队列,咱们只是选择其中的一个直接拿来用。而Create函数是实打实的从头开始去建立一个队列。
2)在iOS6.0以前,在GCD中凡是使用了带Create和retain的函数在最后都须要作一次release操做。而主队列和全局并发队列不须要咱们手动release。固然了,在iOS6.0以后GCD已经被归入到了ARC的内存管理范畴中,即使是使用retain或者create函数建立的对象也再也不须要开发人员手动释放,咱们像对待普通OC对象同样对待GCD就OK。
3)在使用栅栏函数的时候,苹果官方明确规定栅栏函数只有在和使用create函数本身的建立的并发队列一块儿使用的时候才有效(没有给出具体缘由)
4)其它区别涉及到XNU内核的系统级线程编程,不一一列举。
5)给出一些参考资料(能够自行研究):
GCDAPI:https://developer.apple.com/library/ios/documentation/Performance/Reference/GCD_libdispatch_Ref/index.html#//apple_ref/c/func/dispatch_queue_create
Libdispatch版本源码:http://www.opensource.apple.com/source/libdispatch/libdispatch-187.5/
GCD开源:http://libdispatch.macosforge.org
```
=================================================================================html

####2 单例模式
```objc
1 基本概念
1)单例模式
在程序运行过程,一个类只有一个实例
2)使用场合
在整个应用程序中,共享一份资源(这份资源只须要建立初始化1次)ios

2 ARC实现单例
1)步骤
01 在类的内部提供一个static修饰的全局变量
02 提供一个类方法,方便外界访问
03 重写+allocWithZone方法,保证永远都只为单例对象分配一次内存空间
04 严谨起见,重写-copyWithZone方法和-MutableCopyWithZone方法程序员

2)相关代码
//提供一个static修饰的全局变量,强引用着已经实例化的单例对象实例
static XMGTools *_instance;macos

//类方法,返回一个单例对象
+(instancetype)shareTools
{
//注意:这里建议使用self,而不是直接使用类名Tools(考虑继承)编程

return [[self alloc]init];
}缓存

//保证永远只分配一次存储空间
+(instancetype)allocWithZone:(struct _NSZone *)zone
{
//使用GCD中的一次性代码
// static dispatch_once_t onceToken;
// dispatch_once(&onceToken, ^{
// _instance = [super allocWithZone:zone];
// });网络

//使用加锁的方式,保证只分配一次存储空间
@synchronized(self) {
if (_instance == nil) {
_instance = [super allocWithZone:zone];
}
}
return _instance;
}
/*
1. mutableCopy 建立一个新的可变对象,并初始化为原对象的值,新对象的引用计数为 1;
2. copy 返回一个不可变对象。分两种状况:(1)若原对象是不可变对象,那么返回原对象,并将其引用计数加 1 ;(2)若原对象是可变对象,那么建立一个新的不可变对象,并初始化为原对象的值,新对象的引用计数为 1。
*/
//让代码更加的严谨
-(nonnull id)copyWithZone:(nullable NSZone *)zone
{
// return [[self class] allocWithZone:zone];
return _instance;
}数据结构

-(nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
return _instance;
}多线程

3 MRC实现单例
1)实现步骤
01 在类的内部提供一个static修饰的全局变量
02 提供一个类方法,方便外界访问
03 重写+allocWithZone方法,保证永远都只为单例对象分配一次内存空间
04 严谨起见,重写-copyWithZone方法和-MutableCopyWithZone方法
05 重写release方法
06 重写retain方法
07 建议在retainCount方法中返回一个最大值
2)配置MRC环境知识
01 注意ARC不是垃圾回收机制,是编译器特性
02 配置MRC环境:build setting ->搜索automatic ref->修改成NO
3)相关代码
//提供一个static修饰的全局变量,强引用着已经实例化的单例对象实例
static XMGTools *_instance;并发

//类方法,返回一个单例对象
+(instancetype)shareTools
{
//注意:这里建议使用self,而不是直接使用类名Tools(考虑继承)

return [[self alloc]init];
}

//保证永远只分配一次存储空间
+(instancetype)allocWithZone:(struct _NSZone *)zone
{
//使用GCD中的一次性代码
// static dispatch_once_t onceToken;
// dispatch_once(&onceToken, ^{
// _instance = [super allocWithZone:zone];
// });

//使用加锁的方式,保证只分配一次存储空间
@synchronized(self) {
if (_instance == nil) {
_instance = [super allocWithZone:zone];
}
}
return _instance;
}

//让代码更加的严谨
-(nonnull id)copyWithZone:(nullable NSZone *)zone
{
// return [[self class] allocWithZone:zone];
return _instance;
}

-(nonnull id)mutableCopyWithZone:(nullable NSZone *)zone
{
return _instance;
}

//在MRC环境下,若是用户retain了一次,那么直接返回instance变量,不对引用计数器+1
//若是用户release了一次,那么什么都不作,由于单例模式在整个程序运行过程当中都拥有且只有一份,程序退出以后被释放,因此不须要对引用计数器操做
-(oneway void)release
{
}

-(instancetype)retain
{
return _instance;
}

//惯用法,有经验的程序员经过打印retainCount这个值能够猜到这是一个单例
-(NSUInteger)retainCount
{
return MAXFLOAT;
}

4 通用版本
1)有意思的对话
01 问:写一份单例代码在ARC和MRC环境下都适用?
答:可使用条件编译来判断当前项目环境是ARC仍是MRC
02 问:条件编译的代码呢,么么哒?
//答:条件编译
#if __has_feature(objc_arc)
//若是是ARC,那么就执行这里的代码1
#else
//若是不是ARC,那么就执行代理的代码2
#endif
03 问:在项目里面每每须要实现不少的单例,好比下载、网络请求、音乐播放等等,弱弱的问一句单例能够用继承吗?
答:单例是不能够用继承的,若是想一次写就,四处使用,那么推荐亲使用带参数的宏定义啦!
04 问:宏定义怎么弄?
答:这个嘛~~回头看一眼个人代码咯,亲。

2)使用带参数的宏完成通用版单例模式代码
01 注意条件编译的代码不能包含在宏定义里面
02 宏定义的代码只须要写一次就好,以后直接拖到项目中用就OK
=================================================================================
```

####3 NSOperation
(1)NSOperation基本使用
```objc
1)相关概念
01 NSOperation是对GCD的包装
02 两个核心概念【队列+操做】

2)基本使用
01 NSOperation自己是抽象类,只能只有它的子类
02 三个子类分别是:NSBlockOperation、NSInvocationOperation以及自定义继承自NSOperation的类
03 NSOperation和NSOperationQueue结合使用实现多线程并发

3)相关代码
// 01 NSInvocationOperation
//1.封装操做
/*
第一个参数:目标对象
第二个参数:该操做要调用的方法,最多接受一个参数
第三个参数:调用方法传递的参数,若是方法不接受参数,那么该值传nil
*/
NSInvocationOperation *operation = [[NSInvocationOperation alloc]
initWithTarget:self selector:@selector(run) object:nil];

//2.启动操做
[operation start];
---------------
// 02 NSBlockOperation
//1.封装操做
/*
NSBlockOperation提供了一个类方法,在该类方法中封装操做
*/
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
//在主线程中执行
NSLog(@"---download1--%@",[NSThread currentThread]);
}];

//2.追加操做,追加的操做在子线程中执行
[operation addExecutionBlock:^{
NSLog(@"---download2--%@",[NSThread currentThread]);
}];

[operation addExecutionBlock:^{
NSLog(@"---download3--%@",[NSThread currentThread]);
}];

//3.启动执行操做
[operation start];

---------------
// 03 自定义NSOperation
//如何封装操做?
//自定义的NSOperation,经过重写内部的main方法实现封装操做
-(void)main
{
NSLog(@"--main--%@",[NSThread currentThread]);
}

//如何使用?
//1.实例化一个自定义操做对象
XMGOperation *op = [[XMGOperation alloc]init];

//2.执行操做
[op start];
```
(2)NSOperationQueue基本使用
```objc
1)NSOperation中的两种队列
01 主队列 经过mainQueue得到,凡是放到主队列中的任务都将在主线程执行
02 非主队列 直接alloc init出来的队列。非主队列同时具有了并发和串行的功能,经过设置最大并发数属性来控制任务是并发执行仍是串行执行

2)相关代码
//自定义NSOperation
-(void)customOperation
{
//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操做
//好处:1.信息隐蔽
//2.代码复用

XMGOperation *op1 = [[XMGOperation alloc]init];
XMGOperation *op2 = [[XMGOperation alloc]init];

//3.添加操做到队列中
[queue addOperation:op1];
[queue addOperation:op2];
}

//NSBlockOperation
- (void)block
{
//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操做
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"1----%@",[NSThread currentThread]);
}];

NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"2----%@",[NSThread currentThread]);

}];

[op2 addExecutionBlock:^{
NSLog(@"3----%@",[NSThread currentThread]);
}];

[op2 addExecutionBlock:^{
NSLog(@"4----%@",[NSThread currentThread]);
}];

//3.添加操做到队列中
[queue addOperation:op1];
[queue addOperation:op2];

//补充:简便方法
[queue addOperationWithBlock:^{
NSLog(@"5----%@",[NSThread currentThread]);
}];

}

//NSInvocationOperation
- (void)invocation
{
/*
GCD中的队列:
串行队列:本身建立的,主队列
并发队列:本身建立的,全局并发队列

NSOperationQueue
主队列:[NSOperationQueue mainqueue];凡事放在主队列中的操做都在主线程中执行
非主队列:[[NSOperationQueue alloc]init],并发和串行,默认是并发执行的
*/

//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操做
NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download1) object:nil];

NSInvocationOperation *op2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download2) object:nil];


NSInvocationOperation *op3 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download3) object:nil];


//3.把封装好的操做添加到队列中
[queue addOperation:op1];//[op1 start]
[queue addOperation:op2];
[queue addOperation:op3];
}
```
(3) NSOperation其它用法
```objc
1)设置最大并发数【控制任务并发和串行】
//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.设置最大并发数
//注意点:该属性须要在任务添加到队列中以前进行设置
//该属性控制队列是串行执行仍是并发执行
//若是最大并发数等于1,那么该队列是串行的,若是大于1那么是并行的
//系统的最大并发数有个默认的值,为-1,若是该属性设置为0,那么不会执行任何任务
queue.maxConcurrentOperationCount = 2;

2)暂停和恢复以及取消
//设置暂停和恢复
//suspended设置为YES表示暂停,suspended设置为NO表示恢复
//暂停表示不继续执行队列中的下一个任务,暂停操做是能够恢复的
if (self.queue.isSuspended) {
self.queue.suspended = NO;
}else
{
self.queue.suspended = YES;
}

//取消队列里面的全部操做
//取消以后,当前正在执行的操做的下一个操做将再也不执行,并且永远都不在执行,就像后面的全部任务都从队列里面移除了同样
//取消操做是不能够恢复的
[self.queue cancelAllOperations];

---------自定义NSOperation取消操做--------------------------
-(void)main
{
//耗时操做1
for (int i = 0; i<1000; i++) {
NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
}
NSLog(@"+++++++++++++++++++++++++++++++++");

//苹果官方建议,每当执行完一次耗时操做以后,就查看一下当前队列是否为取消状态,若是是,那么就直接退出
//好处是能够提升程序的性能
if (self.isCancelled) {
return;
}

//耗时操做2
for (int i = 0; i<1000; i++) {
NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
}

NSLog(@"+++++++++++++++++++++++++++++++++");
}
```
(4) NSOperation实现线程间通讯
```objc
1)开子线程下载图片
//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.使用简便方法封装操做并添加到队列中
[queue addOperationWithBlock:^{

//3.在该block中下载图片
NSURL *url = [NSURL URLWithString:@"http://news.51sheyuan.com/uploads/allimg/111001/133442IB-2.jpg"];
NSData *data = [NSData dataWithContentsOfURL:url];
UIImage *image = [UIImage imageWithData:data];
NSLog(@"下载图片操做--%@",[NSThread currentThread]);

//4.回到主线程刷新UI
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
self.imageView.image = image;
NSLog(@"刷新UI操做---%@",[NSThread currentThread]);
}];
}];

2)下载多张图片合成综合案例(设置操做依赖)
//02 综合案例
- (void)download2
{
NSLog(@"----");
//1.建立队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操做下载图片1
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{

NSURL *url = [NSURL URLWithString:@"http://h.hiphotos.baidu.com/zhidao/pic/item/6a63f6246b600c3320b14bb3184c510fd8f9a185.jpg"];
NSData *data = [NSData dataWithContentsOfURL:url];

//拿到图片数据
self.image1 = [UIImage imageWithData:data];
}];


//3.封装操做下载图片2
NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
NSURL *url = [NSURL URLWithString:@"http://pic.58pic.com/58pic/13/87/82/27Q58PICYje_1024.jpg"];
NSData *data = [NSData dataWithContentsOfURL:url];

//拿到图片数据
self.image2 = [UIImage imageWithData:data];
}];

//4.合成图片
NSBlockOperation *combine = [NSBlockOperation blockOperationWithBlock:^{

//4.1 开启图形上下文
UIGraphicsBeginImageContext(CGSizeMake(200, 200));

//4.2 画image1
[self.image1 drawInRect:CGRectMake(0, 0, 200, 100)];

//4.3 画image2
[self.image2 drawInRect:CGRectMake(0, 100, 200, 100)];

//4.4 根据图形上下文拿到图片数据
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
// NSLog(@"%@",image);

//4.5 关闭图形上下文
UIGraphicsEndImageContext();

//7.回到主线程刷新UI
[[NSOperationQueue mainQueue]addOperationWithBlock:^{
self.imageView.image = image;
NSLog(@"刷新UI---%@",[NSThread currentThread]);
}];

}];

//5.设置操做依赖
[combine addDependency:op1];
[combine addDependency:op2];

//6.添加操做到队列中执行
[queue addOperation:op1];
[queue addOperation:op2];
[queue addOperation:combine];
}
```
(5) CD和NSOperation的对比:
```objc
1)GCD是纯C语言的API,而操做队列则是Object-C的对象。
2)在GCD中,任务用块(block)来表示,而块是个轻量级的数据结构;相反操做队列中的『操做』NSOperation则是个更加剧量级的Object-C对象。
3)具体该使用GCD仍是使用NSOperation须要看具体的状况

NSOperation和NSOperationQueue相对GCD的好处有:
1)NSOperationQueue能够方便的调用cancel方法来取消某个操做,而GCD中的任务是没法被取消的(安排好任务以后就无论了)。
2)NSOperation能够方便的指定操做间的依赖关系。
3)NSOperation能够经过KVO提供对NSOperation对象的精细控制(如监听当前操做是否被取消或是否已经完成等)
4)NSOperation能够方便的指定操做优先级。操做优先级表示此操做与队列中其它操做之间的优先关系,优先级高的操做先执行,优先级低的后执行。
5)经过自定义NSOperation的子类能够实现操做重用,
```
=================================================================================

####4 多图下载综合示例程序
```objc
1 涉及知识点
01 字典转模型
02 图片重复下载---》内存缓存,沙盒缓存处理
03 UI不流畅---》开子线程下载图片(注意线程间通讯)
04 图片下载任务被添加到队列中屡次---》操做缓存处理
05 图片下载后不显示问题---》主动刷新指定行
06 图片加载中出现数据错乱问题---》设置占位图片
07 在程序开发过程当中的一些容错处理
```
=================================================================================

####5 第三方框架
```objc
1 SDWebImage基本使用(设置imageView的图片)
[cell.imageView sd_setImageWithURL:[NSURL URLWithString:app.icon] placeholderImage:[UIImage imageNamed:@"placehoder"]];

2 SDWebImage内部结构```

相关文章
相关标签/搜索