猫猫学iOS(五十一)多线程网络之GCD下载合并图片_队列组的使用

本文主要是介绍猫猫学iOS(五十一)多线程网络之GCD下载合并图片_队列组的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

猫猫分享,必须精品

原创文章,欢迎转载。转载请注明:翟乃玉的博客
地址:http://blog.csdn.net/u013357243?viewmode=contents

合并图片(图片水印)第一种方法

效果

这里写图片描述

实现:

思路:
1.分别下载2张图片:大图片、LOGO
2.合并2张图片
3.显示到一个imageView身上

  // 异步下载dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 1.下载第1张NSURL *url1 = [NSURL URLWithString:@"http://g.hiphotos.baidu.com/image/pic/item/f2deb48f8c5494ee460de6182ff5e0fe99257e80.jpg"];NSData *data1 = [NSData dataWithContentsOfURL:url1];UIImage *image1 = [UIImage imageWithData:data1];// 2.下载第2张NSURL *url2 = [NSURL URLWithString:@"http://su.bdimg.com/static/superplus/img/logo_white_ee663702.png"];NSData *data2 = [NSData dataWithContentsOfURL:url2];UIImage *image2 = [UIImage imageWithData:data2];// 3.合并图片// 开启一个位图上下文UIGraphicsBeginImageContextWithOptions(image1.size, NO, 0.0);// 绘制第1张图片CGFloat image1W = image1.size.width;CGFloat image1H = image1.size.height;[image1 drawInRect:CGRectMake(0, 0, image1W, image1H)];// 绘制第2张图片CGFloat image2W = image2.size.width * 0.5;CGFloat image2H = image2.size.height * 0.5;CGFloat image2Y = image1H - image2H;[image2 drawInRect:CGRectMake(0, image2Y, image2W, image2H)];// 得到上下文中的图片UIImage *fullImage = UIGraphicsGetImageFromCurrentImageContext();// 结束上下文UIGraphicsEndImageContext();// 4.回到主线程显示图片dispatch_async(dispatch_get_main_queue(), ^{self.imageView.image = fullImage;});});

但是这样虽然在异步线程中完成的,不过,当下载完第一个图的时候,第二个图还没下,在等待,我们想让每一个图都在一个单独的异步线程里面下载,于是我们用第二种方法。


:第二种方法

首先定义两个imageView

@property (weak, nonatomic) IBOutlet UIImageView *imageView;
@property (nonatomic, strong) UIImage *image1;
@property (nonatomic, strong) UIImage *image2;

然后实现这样实现

 // 异步下载dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 1.下载第1张NSURL *url1 = [NSURL URLWithString:@"http://g.hiphotos.baidu.com/image/pic/item/f2deb48f8c5494ee460de6182ff5e0fe99257e80.jpg"];NSData *data1 = [NSData dataWithContentsOfURL:url1];self.image1 = [UIImage imageWithData:data1];[self bindImages];});dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 2.下载第2张NSURL *url2 = [NSURL URLWithString:@"http://su.bdimg.com/static/superplus/img/logo_white_ee663702.png"];NSData *data2 = [NSData dataWithContentsOfURL:url2];self.image2 = [UIImage imageWithData:data2];[self bindImages];});

工作原理是开两个异步线程来分别下载两个图片,下载结束后都会调用
【self bindImages】方法,这时候我们在s 方法中设置回到主界面刷新,不过首先要判断两张图是否有数据,如下:

- (void)bindImages
{if (self.image1 == nil || self.image2 == nil) return;// 3.合并图片// 开启一个位图上下文UIGraphicsBeginImageContextWithOptions(self.image1.size, NO, 0.0);// 绘制第1张图片CGFloat image1W = self.image1.size.width;CGFloat image1H = self.image1.size.height;[self.image1 drawInRect:CGRectMake(0, 0, image1W, image1H)];// 绘制第2张图片CGFloat image2W = self.image2.size.width * 0.5;CGFloat image2H = self.image2.size.height * 0.5;CGFloat image2Y = image1H - image2H;[self.image2 drawInRect:CGRectMake(0, image2Y, image2W, image2H)];// 得到上下文中的图片UIImage *fullImage = UIGraphicsGetImageFromCurrentImageContext();// 结束上下文UIGraphicsEndImageContext();// 4.回到主线程显示图片dispatch_async(dispatch_get_main_queue(), ^{self.imageView.image = fullImage;});
}

虽然我们通过我们自己的想法解决了问题,不过其实我们有更好的解决方案,那就是队列组


第三种方法:队列组完成

什么时候用队列组呢?

首先:分别异步执行2个耗时的操作
其次:等2个异步操作都执行完毕后,再回到主线程执行操作

如果想要快速高效地实现上述需求,可以考虑用队列组,用法如下:

// 1.创建队列组
dispatch_group_t group =  dispatch_group_create();
// 2.1第一个队列组异步
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 执行1个耗时的异步操作
});// 2.2第二个队列组异步
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 执行1个耗时的异步操作
});// 3.所有队列组异步线程结束后
dispatch_group_notify(group, dispatch_get_main_queue(), ^{// 等前面的异步操作都执行完毕后,回到主线程...
});

那就让我们来用队列组完成他吧,代码如下:

// 1.队列组dispatch_group_t group = dispatch_group_create();dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);// 2.下载图片1__block UIImage *image1 = nil;dispatch_group_async(group, queue, ^{NSURL *url1 = [NSURL URLWithString:@"http://g.hiphotos.baidu.com/image/pic/item/f2deb48f8c5494ee460de6182ff5e0fe99257e80.jpg"];NSData *data1 = [NSData dataWithContentsOfURL:url1];image1 = [UIImage imageWithData:data1];});// 3.下载图片2__block UIImage *image2 = nil;dispatch_group_async(group, queue, ^{NSURL *url2 = [NSURL URLWithString:@"http://su.bdimg.com/static/superplus/img/logo_white_ee663702.png"];NSData *data2 = [NSData dataWithContentsOfURL:url2];image2 = [UIImage imageWithData:data2];});// 4.合并图片 (保证执行完组里面的所有任务之后,再执行notify函数里面的block)dispatch_group_notify(group, queue, ^{// 开启一个位图上下文UIGraphicsBeginImageContextWithOptions(image1.size, NO, 0.0);// 绘制第1张图片CGFloat image1W = image1.size.width;CGFloat image1H = image1.size.height;[image1 drawInRect:CGRectMake(0, 0, image1W, image1H)];// 绘制第2张图片CGFloat image2W = image2.size.width * 0.5;CGFloat image2H = image2.size.height * 0.5;CGFloat image2Y = image1H - image2H;[image2 drawInRect:CGRectMake(0, image2Y, image2W, image2H)];// 得到上下文中的图片UIImage *fullImage = UIGraphicsGetImageFromCurrentImageContext();// 结束上下文UIGraphicsEndImageContext();// 5.回到主线程显示图片dispatch_async(dispatch_get_main_queue(), ^{self.imageView.image = fullImage;});});

__block关键字

注意:这里我们用到了一个__block的关键字,这是因为如果不加的话,block里面的是不能访问block外面的东西的。

小小总结:

开启异步

(直接敲dispatch_a。。。就能出来一堆提示,然后选选填填,没啥难度,前面说过了)

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{code
});

开启队列组:

首先建一个队列组:

dispatch_group_t group =  dispatch_group_create();

然后把原来放队列的地方换成队列组 并且用dispatch_get_global_queue这个

dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 执行1个耗时的异步操作
});

上面这段能多弄几个,弄几个就有几个线程

然后结束时候用这个:

dispatch_group_notify(group, dispatch_get_main_queue(), ^{// 等前面的异步操作都执行完毕后,回到主线程...
});

合并图片的过程

步骤:
1.开启一个位图上下文
2.1绘制第1张图片
2.2绘制第2张图片
3.得到上下文中的图片
4.结束上下文

// 1.开启一个位图上下文UIGraphicsBeginImageContextWithOptions(image1.size, NO, 0.0);// 2.1绘制第1张图片CGFloat image1W = image1.size.width;CGFloat image1H = image1.size.height;[image1 drawInRect:CGRectMake(0, 0, image1W, image1H)];// 2.2绘制第2张图片CGFloat image2W = image2.size.width * 0.5;CGFloat image2H = image2.size.height * 0.5;CGFloat image2Y = image1H - image2H;[image2 drawInRect:CGRectMake(0, image2Y, image2W, image2H)];// 3.得到上下文中的图片UIImage *fullImage = UIGraphicsGetImageFromCurrentImageContext();// 4.结束上下文UIGraphicsEndImageContext();

这篇关于猫猫学iOS(五十一)多线程网络之GCD下载合并图片_队列组的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138349

相关文章

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

使用Python实现图像LBP特征提取的操作方法

《使用Python实现图像LBP特征提取的操作方法》LBP特征叫做局部二值模式,常用于纹理特征提取,并在纹理分类中具有较强的区分能力,本文给大家介绍了如何使用Python实现图像LBP特征提取的操作方... 目录一、LBP特征介绍二、LBP特征描述三、一些改进版本的LBP1.圆形LBP算子2.旋转不变的LB

Maven的使用和配置国内源的保姆级教程

《Maven的使用和配置国内源的保姆级教程》Maven是⼀个项目管理工具,基于POM(ProjectObjectModel,项目对象模型)的概念,Maven可以通过一小段描述信息来管理项目的构建,报告... 目录1. 什么是Maven?2.创建⼀个Maven项目3.Maven 核心功能4.使用Maven H

Redis消息队列实现异步秒杀功能

《Redis消息队列实现异步秒杀功能》在高并发场景下,为了提高秒杀业务的性能,可将部分工作交给Redis处理,并通过异步方式执行,Redis提供了多种数据结构来实现消息队列,总结三种,本文详细介绍Re... 目录1 Redis消息队列1.1 List 结构1.2 Pub/Sub 模式1.3 Stream 结

Python中__init__方法使用的深度解析

《Python中__init__方法使用的深度解析》在Python的面向对象编程(OOP)体系中,__init__方法如同建造房屋时的奠基仪式——它定义了对象诞生时的初始状态,下面我们就来深入了解下_... 目录一、__init__的基因图谱二、初始化过程的魔法时刻继承链中的初始化顺序self参数的奥秘默认

SpringBoot使用GZIP压缩反回数据问题

《SpringBoot使用GZIP压缩反回数据问题》:本文主要介绍SpringBoot使用GZIP压缩反回数据问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot使用GZIP压缩反回数据1、初识gzip2、gzip是什么,可以干什么?3、Spr

Spring Boot 集成 Quartz并使用Cron 表达式实现定时任务

《SpringBoot集成Quartz并使用Cron表达式实现定时任务》本篇文章介绍了如何在SpringBoot中集成Quartz进行定时任务调度,并通过Cron表达式控制任务... 目录前言1. 添加 Quartz 依赖2. 创建 Quartz 任务3. 配置 Quartz 任务调度4. 启动 Sprin

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停