iOS 下利用 CGContextRef 简单实现多图拼接功能

2023-10-24 21:20

本文主要是介绍iOS 下利用 CGContextRef 简单实现多图拼接功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

👇👇关注后回复 “进群” ,拉你进程序员交流群👇👇

废话开篇:简单实现多张小图拼接为一张长整图的功能。

一、需要实现的功能

将这三张 酆都大帝 拼接成下面的一张整图

outside_default.png

最终要保存的图

outside_default.png outside_default.png
酆都大帝.jpg

二、合并功能操作代码

在控制器里进行多图的拼接操作,

UIImage * image1 = [UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"fd1" ofType:@"jpeg"]];UIImage * image2 = [UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"fd2" ofType:@"jpeg"]];UIImage * image3 = [UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"fd3" ofType:@"jpeg"]];self.scrollView = [[UIScrollView alloc] initWithFrame:self.view.bounds];[self.view addSubview:self.scrollView];//合并小图[WSLImageCombineOperation combineImages:@[image1,image2,image3] callBack:^(UIImage * _Nonnull resultImage) {UIImageView * imageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, self.scrollView.frame.size.height *  resultImage.size.width / resultImage.size.height,self.scrollView.frame.size.height)];imageView.image = resultImage;imageView.contentMode = UIViewContentModeScaleAspectFit;[self.scrollView addSubview:imageView];self.scrollView.contentSize = CGSizeMake(imageView.frame.size.width, imageView.frame.size.height);//保存到沙盒[WSLImageCombineOperation saveImageToCache:resultImage];}];
复制代码

三、WSLImageCombineOperation 类,实现图合并功能

其实这里主要的是利用 CGContextRef 上下文进行的绘制。在开辟上下文后,在上下文里面进行多图的绘制,最后将上下文内容转为 image 进行展示与保存。

在开辟上下文的之前,需要做几件事:

1、创建 CGContextRef 上下文。

创建上下文之前,需要计算多张图上下依次拼接的最大高度图组中最宽的一张图片宽度

保证上下文满足两个条件:

(1)足够的高度。

(2)以最大图片为准,居中排列。

//同时获取图片组最大宽度及拼接高度
+ (void)getAllImagesMaxWidthAndTotleHeight:(NSArray *)images callBack:(void(^)(NSUInteger maxWith,NSUInteger totleHeight))callBack
{CGFloat maxWidth = 0;CGFloat allImageHeight = 0;for (UIImage * image in images) {CGFloat currentImageWidth = [self imageWidth:image];maxWidth = maxWidth < currentImageWidth ? currentImageWidth : maxWidth;allImageHeight += [self imageHeight:image];}callBack(maxWidth,allImageHeight);
}//获取图片宽度
+ (CGFloat)imageWidth:(UIImage *)image
{return image.size.width;
}//获取图片高度
+ (CGFloat)imageHeight:(UIImage *)image
{return image.size.height;
}
复制代码
2、绘制内容到 CGContextRef 上下文中

在绘制前,可以先设置上下的填充颜色。

//设置大图上下文背景填充颜色(这里是黑色)CGContextSetFillColorWithColor(contextRef, [UIColor blackColor].CGColor);//设置大图上下文背景填充区域(全部区域)CGContextFillRect(contextRef, CGRectMake(0, 0, imageWidth, imageHeight));
复制代码

其次,进行循环绘制,直接放完整代码

+ (void)combineImages:(NSArray *)images callBack:(void(^)(UIImage * resultImage))callBack
{//获取拼接后图片的高度及最大宽度[self getAllImagesMaxWidthAndTotleHeight:images callBack:^(NSUInteger imageWidth, NSUInteger imageHeight) {//创建颜色空间CGColorSpaceRef colorSpaceRef = CGColorSpaceCreateDeviceRGB();//开辟像素区域UInt32 *imagePiexl = (UInt32 *)calloc(imageWidth*imageHeight, sizeof(UInt32));//创建上下文CGContextRef contextRef = CGBitmapContextCreate(imagePiexl,imageWidth,imageHeight,8,4*imageWidth,colorSpaceRef,kCGImageAlphaNoneSkipLast);//设置大图上下文背景填充颜色(这里是黑色)CGContextSetFillColorWithColor(contextRef, [UIColor blackColor].CGColor);//设置大图上下文背景填充区域(全部区域)CGContextFillRect(contextRef, CGRectMake(0, 0, imageWidth, imageHeight));//拼接图片 X 值NSUInteger imageX = 0;//拼接图片 Y 值NSUInteger imageY = 0//循环遍历进行绘制for (UIImage * image in images) {//循环绘制CGImageRef imageRef = image.CGImage;NSUInteger width = CGImageGetWidth(imageRef);NSUInteger height = CGImageGetHeight(imageRef);//谢谢[可爱亲宝宝]的纠正imageX = (imageWidth - width) / 2.0;CGContextDrawImage(contextRef, CGRectMake(imageX, imageY, width, height), imageRef);//重新计算 Y 值imageY += height;}//生成图片,供外部 UIImageView 渲染CGImageRef resultRef = CGBitmapContextCreateImage(contextRef);UIImage * resultImage = [UIImage imageWithCGImage:resultRef scale:[UIScreen mainScreen].scale orientation:UIImageOrientationUp];callBack(resultImage);//释放用过的内存CGContextRelease(contextRef);CGColorSpaceRelease(colorSpaceRef);free(imagePiexl);CGImageRelease(resultRef);}];}
复制代码
3、保存图片到沙盒

实现图片保存到沙盒的功能

//保存到沙盒
+ (BOOL)saveImageToCache:(UIImage *)image
{NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask, YES);NSString *filePath = [[paths objectAtIndex:0] stringByAppendingPathComponent:@"酆都大帝.jpg"];NSLog(@"filePath = %@",filePath);BOOL result = [UIImagePNGRepresentation(image) writeToFile: filePath atomically:YES];return result;
}
复制代码

这里打印 filePath 缓存路径,复制出来,直接在 MAC电脑 “前往文件夹”下输入此路径就能找到模拟器下的对应沙盒文件。

outside_default.png

四、总结与思考

上述实现原理比较简单。除此之外,如果是要某张小图的某一部分或者对图片整体像素处理,那么,就需要对图片进行像素操作了。个人理解,通过 image.CGImage 可以获得图片的很多信息数据,图片格式、图片宽高、三色素值等具体的信息,只要最后修改的文件符合图片格式,那么,其他的操作就都是二进制数据的修改与拼接。

代码拙劣,大神勿笑,互相讨论,共同进步[抱拳][抱拳][抱拳]。

转自:掘金-头痛脑胀的代码……

链接:https://juejin.cn/post/7048895249128620069

-End-

最近有一些小伙伴,让我帮忙找一些 面试题 资料,于是我翻遍了收藏的 5T 资料后,汇总整理出来,可以说是程序员面试必备!所有资料都整理到网盘了,欢迎下载!

47265c69c4b6ad7aa0b4d382c7533004.png

点击👆卡片,关注后回复【面试题】即可获取

在看点这里75b614c72633e7cae8fb09beded74993.gif好文分享给更多人↓↓

这篇关于iOS 下利用 CGContextRef 简单实现多图拼接功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand