多线程GCD geekband

GCD
Grand Central Dispatch

GCD中也有一个类似于NSOperationQueue的队列,GCD统一管理整个队列中的任务。但是GCD中的队列分为并行队列和串行队列两类:

串行队列:只有一个线程,加入到队列中的操作按添加顺序依次执行。
串行队列:有多个线程,操作进来之后它会将这些队列安排在可用的处理器上,同时保证先进来的任务优先处理。

GCD中有2个用来执行任务的函数
只是根据队列的性质,分为<1>串行队列:用户队列、主线程队列 <2>并行队列.
说明:把右边的参数(任务)提交给左边的参数(队列)进行执行。

 (1)用串行队列的方式执行任务 dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);

参数说明:

queue:队列

block:任务

同步(dispatch_sync)、异步方式(dispatch_async). 配合串行队列和并行队列使用.

效果显示.
异步串行


如果图挂了 点击超链接

异步并行


如果图挂了 点击超链接
同步

如果图挂了 点击超链接
我相信这里gif的展示令你更深刻认识到GCD.
其他任务执行方法


贴上案例

#import "ViewController.h"

@interface ViewController ()
@property (strong , nonatomic) NSMutableArray *imageView;
@property  (strong,nonatomic)   NSArray *imageNames;
@end

@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.

//数组
NSString *Photo1 =[NSString stringWithFormat:@"http://news.mydrivers.com/img/20130518/3fed7f3e8ab848e284bb2238320cbd93.jpg"];
NSString *Photo2 =[NSString stringWithFormat:@"http://image.tianjimedia.com/uploadImages/2013/237/015W189B5F99_B75.JPG"];
NSString *Photo3 =[NSString stringWithFormat:@"http://pic.3h3.com/up/2014-6/20146614141118561372.jpg"];
NSString *Photo4 =[NSString stringWithFormat:@"http://www.3dmgame.com/uploads/allimg/130518/154_130518143149_1.jpg"];
NSString *Photo5 =[NSString stringWithFormat:@"http://b.hiphotos.baidu.com/zhidao/pic/item/cdbf6c81800a19d8a50020aa31fa828ba61e4619.jpg"];
NSString *Photo6 =[NSString stringWithFormat:@"http://img8.zol.com.cn/bbs/upload/23416/23415166.jpg"];
self.imageNames =[NSArray arrayWithObjects:
                 Photo1,Photo2,Photo3, Photo4,Photo5,Photo6,nil];

[self loadScrollViewWithButton];

}




-(void)loadimageViews{
//ImageView
_imageView=[NSMutableArray array];
for (int i = 0 ; i <6 ;  i++) {
    UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(5 +(self.view.bounds.size.width/2 )*(i/3),(self.view.bounds.size.height/3 -25)*(i%3), self.view.bounds.size.width/2-10, self.view.bounds.size.width/2-10)];
    imageView.backgroundColor = [UIColor redColor];
    [self.view addSubview:imageView];
    [_imageView addObject:imageView];
    
}
}
-(void)loadScrollViewWithButton{
UIScrollView *scrollView =[[UIScrollView alloc]initWithFrame:CGRectMake(0, self.view.bounds.size.height -100, self.view.bounds.size.width, 40)];
scrollView.backgroundColor = [UIColor lightGrayColor];    scrollView.alpha = 0.6;
scrollView.contentSize = CGSizeMake(1000, 0);
[self.view addSubview:scrollView];

UILabel *lable = [[UILabel alloc]initWithFrame:CGRectMake(0, 0, 1000, 60)];
lable.text = @"scrollView测试滚动,加载时滚动条体验.图片加载是否在主线程中";
[scrollView addSubview:lable];
//串行队列
UIButton *TandemQueueButton=[UIButton buttonWithType:UIButtonTypeRoundedRect];
TandemQueueButton.frame=CGRectMake(0, self.view.bounds.size.height -60, 100,60);
[TandemQueueButton setTitle:@"异步串行队列" forState:UIControlStateNormal];
TandemQueueButton.backgroundColor =[UIColor grayColor];
TandemQueueButton.alpha = 0.8;
[TandemQueueButton addTarget:self action:@selector(loadAsyncTandemQueue) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:TandemQueueButton];

//异步串行队列
UIButton *button=[UIButton buttonWithType:UIButtonTypeRoundedRect];
button.frame=CGRectMake(110, self.view.bounds.size.height -60, 100,60);
[button setTitle:@"异步并行队列" forState:UIControlStateNormal];
button.backgroundColor =[UIColor grayColor];
button.alpha = 0.8;

[button addTarget:self action:@selector(loadAsyncParallelQueue) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:button];

[TandemQueueButton addTarget:self action:@selector(loadAsyncTandemQueue) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:TandemQueueButton];

//同步队列
UIButton *SynchronousButton=[UIButton buttonWithType:UIButtonTypeRoundedRect];
SynchronousButton.frame=CGRectMake(220, self.view.bounds.size.height -60, 100,60);
[SynchronousButton setTitle:@"同步队列" forState:UIControlStateNormal];
SynchronousButton.backgroundColor =[UIColor grayColor];
SynchronousButton.alpha = 0.8;
[SynchronousButton addTarget:self action:@selector(loadsyncParallelQueue) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:SynchronousButton];

//计事件队列
UIButton *timeButton=[UIButton buttonWithType:UIButtonTypeRoundedRect];
timeButton.frame=CGRectMake(330, self.view.bounds.size.height -60, 100,60);
[timeButton setTitle:@"延迟" forState:UIControlStateNormal];
timeButton.backgroundColor =[UIColor grayColor];
timeButton.alpha = 0.8;
[timeButton addTarget:self action:@selector(loadTimeQueue) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:timeButton];

}

#pragma mark 将图片显示到界面
-(void)updateImageWithData:(NSData *)data andIndex:(int )index{
UIImage *image=[UIImage imageWithData:data];
UIImageView *imageView= _imageView[index];
imageView.image=image;
}

#pragma mark 请求图片数据
-(NSData *)requestData:(int)index{
NSURL *url=[NSURL URLWithString:_imageNames[index]];
NSData *data=[NSData dataWithContentsOfURL:url];
return data;
}
//图片加载
-(void)loadImage:(NSNumber *)index{
int i = (int)[index integerValue];
NSData * data = [self requestData:i];
dispatch_queue_t mainQue = dispatch_get_main_queue()    ;



//为什么要用async?
//async 在主线程中  创建了一个异步线程 加入  全局并发队列,async 不会等待block 执行完成,立即返回

// 1,async 立即返回, viewDidLoad 执行完毕,及主线程执行完毕。
//2,同时,全局并发队列立即执行异步 block , 打印 1, 当执行到 sync 它会等待 block 执行完成才返回, 及等待
//dispatch_get_main_queue() 队列中的 mianThread 执行完成, 然后才开始调用block 。

//体现一下卡顿  sync的卡顿.  他会在在updateImageWithData 卡顿了.因为没有返回;
//    dispatch_async(mainQue, ^{
//        [self updateImageWithData:data andIndex:i];
//    });

dispatch_async(mainQue, ^{
    [self updateImageWithData:data andIndex:i];
});
}
//清除子view
-(void)cleanimage{
[_imageView makeObjectsPerformSelector:@selector(removeFromSuperview)];
}

//异步串行队列意思是一张张按顺序图片加载,
-(void)loadAsyncTandemQueue{
[self cleanimage];
[self loadimageViews];

int  iamgecount = 6;
dispatch_queue_t serialQueue = dispatch_queue_create("myThreadQueue1", DISPATCH_QUEUE_SERIAL);//注意queue对象不是指针类型
//创建多个线程用于填充图片
for (int i=0; i<iamgecount; ++i) {
    //异步执行队列任务
    dispatch_async(serialQueue, ^{
        [self loadImage:[NSNumber numberWithInt:i]];
    });
  }
}
//异步并行队列 不会按顺序加载
-(void)loadAsyncParallelQueue{
[self cleanimage];

[self loadimageViews];
int  iamgecount = 6;
/*dispatch_get_global_queue(参数1, 参数2);
 取得全局队列
 第一个参数:线程优先级
 第二个参数:标记参数,目前没有用,一般传入0
 */
dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
//创建多个线程用于填充图片
for (int i=0; i<iamgecount; ++i) {
    //异步执行队列任务
    dispatch_async(globalQueue, ^{
        [self loadImage:[NSNumber numberWithInt:i]];
    });
}
}

//    GCD  sync 卡顿.
//    内嵌套一个async 就可以运行起来.

//同步队列. 同步 无论并行还是串行 也是最后统一加载
//同时会占据主线程,  造成scrollView 不能拖动
-(void)loadsyncParallelQueue{
[self cleanimage];

[self loadimageViews];
int  iamgecount = 6;



dispatch_queue_t syncQueue = dispatch_queue_create("myThreadQueue2", DISPATCH_QUEUE_SERIAL);
//创建多个线程用于填充图片
for (int i=0; i<iamgecount; ++i) {
   
    //同步执行队列任务
    dispatch_sync(syncQueue, ^{
        NSLog(@"%@ i = %d",syncQueue,i);
         [self loadImage:[NSNumber numberWithInt:i]];
    });

}
}

-(void)loadTimeQueue{

[self cleanimage];
[self loadimageViews];

int  iamgecount = 6;
dispatch_queue_t serialQueue = dispatch_queue_create("myThreadQueue1", DISPATCH_QUEUE_SERIAL);//注意queue对象不是指针类型
//创建多个线程用于填充图片
for (int i=0; i<iamgecount; ++i) {
    //异步执行队列任务
    dispatch_async(serialQueue, ^{
        [self  timeAfter];
        [self loadImage:[NSNumber numberWithInt:i]];
        
    });
}
}


-(void)timeAfter{
dispatch_time_t time = dispatch_time(DISPATCH_TIME_NOW, 10);

dispatch_after(time, dispatch_get_main_queue(), ^{

            NSLog(@"10秒后加载一行图片");
});
}


@end

GCD执行任务的方法并非只有简单的同步调用方法和异步调用方法,还有其他一些常用方法:

 dispatch_apply():重复执行某个任务,但是注意这个方法没有办法异步执行(为了不阻塞线程可以使用dispatch_async()包装一下再执行)。 
 dispatch_once():单次执行一个任务,此方法中的任务只会执行一次,重复调用也没办法重复执行(单例模式中常用此方法)。 
 dispatch_time():延迟一定的时间后执行。 
 dispatch_group_async():实现对任务分组管理,如果一组任务全部完成可以通过dispatch_group_notify()方法获得完成通知(需要定义dispatch_group_t作为分组标识)。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容