写点什么

内存占用过高怎么办?iOS 图片内存优化指南

  • 2020-09-25
  • 本文字数:3459 字

    阅读完需:约 11 分钟

内存占用过高怎么办?iOS图片内存优化指南

导语 | 一般来说,在 App 的内存占用中,图片很容易成为其中的大头。特别是在图片相关的 App 中,稍不注意就容易引发内存占用过高的问题。本文将就 iOS 图片类应用的内存优化展开讨论,希望与大家一同交流。文章作者:张恒铭,腾讯终端开发工程师。

一、内存优化的必要性

事实上,因为目前 iPhone 配备的内存越来越高,当内存占用过高时,并不一定会超过系统设定的阈值而引发强杀进程。


但这并不意味着减少内存占用是没有意义的,因为当内存占用过高时,很容易引起一系列的副作用。最直接的表现是 App Crash,当然还有很多更为深远的副作用。

1. FOOM

FOOM 是最直接的影响了,当内存占用过多导致整个系统的可用内存不足时,App 所在的进程容易被杀掉。而且相比于一般的 Crash 来说,FOOM 更难以检测,并且也更难排查。

2. 限制并发数量

如果一个任务占用了过多的内存,但总的内存是有限的,那么任务的并发数将会受到直接限制。表现上就是 App 里某个功能可同时执行的数量有限,或者可以同时显示的内容有数量限制。


同时,因为内存是有限资源,当占用内存过多时,会容易导致操作系统杀掉其它 App 的进程来给当前的 App 提供足够的内存空间,这对用户体验是不利的。

3. 增加耗电

由于 iOS 系统的 Memory Compressor 的存在,当可用内存不足时,一部分 Dirty Page 会被压缩存储到磁盘中,当用到这部分内存时,再从磁盘里加载回来。这会造成 CPU 花费更多的时间来等待 IO, 间接提高 CPU 占用率,造成耗电。

二、原因分析

1. 图片显示原理

图片其实是由很多个像素点组成的,每个像素点描述了该点的颜色信息。这样的数据是可以被直接渲染在屏幕上的,称之为 Image Buffer。


事实上,由于图片源文件占用的存储空间非常大,一般在存储时候都会进行压缩,非常常见的就是 JPEG 和 PNG 算法压缩的图片。


因此当图片存储在硬盘中的时候,它是经过压缩后的数据。经过解码后的数据才能用于渲染,因此需要将图片显示在屏幕上的话,需要先经过解码。解码后的数据就是 Image Buffer 。



当图片显示在屏幕上时,会复制显示区域的 Image Buffer 去进行渲染。

2. 图片真实占用内存

对于一张正在显示在屏幕上的,尺寸为 1920*1080 的图片来说,如果采用 SRGB 的格式(每个像素点的颜色由 red,green,blue,alpha 一个共 4 个 bytes 来决定)的话,那么它占用的内存为:


1920 * 1080 * 4 = 829440 bytes
复制代码


也就是说,一张非常普通的图片,解码后占用的内存就是 7.9 MB,这是非常夸张的。而图片显示时所占的内存大小是与尺寸和颜色空间正相关的,与压缩算法、图片格式、图片文件的大小没有关联。

三、解决方式

1. 避免将图片放在内存里

对于不显示在屏幕上的图片,在绝大部分时间里,其实是没有必要放在内存里的。解码后的 UIImage 是非常大的,对于不需要显示的图片是不需要解码的。而对于不显示在屏幕上的图片,一般也没有必要继续持有着 UIImage 对象。

2. 图片缩放

图片缩放是很常见的处理方式,一般来说,常见的思想可能是重新画一张小一点的图片,往往是用 UIGraphicsBeginImageContextWithOptions 的方式:


extension UIImage {        public func scaling(to size:CGSize) -> UIImage? {            let drawScale = self.scale            UIGraphicsBeginImageContextWithOptions(size, false, drawScale)            let drawRect:CGRect = CGRect(origin:.zero,size:size)            draw(in: drawRect)            let result = UIGraphicsGetImageFromCurrentImageContext()            UIGraphicsEndImageContext()            return result        }    }
复制代码


这种方式存在以下问题:


第一,默认是 SRGB 的格式,也就是说每个像素需要占 4 个 bytes 的空间,对于一些黑白或者仅有 alpha 通道的数据来说是没有必要的。


第二,需要将原图片完全解码后渲染出来,原图片的解码会造成内存占用的高峰。


对于问题一的解决,可以使用新的 UIGraphicsImageRenderer 的方式,这种情况下框架会自动帮你选择对应的颜色格式,减少不必要的消耗。


extension UIImage {    func scaling(to size:CGSize) -> UIImage? {        let renderer = UIGraphicsImageRenderer(bounds: CGRect(origin: .zero, size: size))        return renderer.image { context in            self.draw(in: context.format.bounds)        }    }}
复制代码


这种方式在一定的场景有所优化,但是没有解决问题二中存在的内存峰值的问题。由于处理前的图片并不一定展示在屏幕上,解码后的数据是冗余信息,因此应该避免图片的解码。


对于峰值过高的问题,最直接的思想是采用流式的方式进行处理。而底层的 ImageIO 的接口就采用了这种方式:


func resizedCgImage(url:URL,for size: CGSize) -> CGImage? {        let options: [CFString: Any] = [            kCGImageSourceShouldCache:false,            kCGImageSourceCreateThumbnailFromImageAlways: true,            kCGImageSourceCreateThumbnailWithTransform: true,            kCGImageSourceShouldCacheImmediately: true,            kCGImageSourceThumbnailMaxPixelSize: max(size.width, size.height)        ]                guard let imageSource = CGImageSourceCreateWithURL(url as NSURL, nil),            let image = CGImageSourceCreateThumbnailAtIndex(imageSource, 0, options as CFDictionary)            else {                                return nil        }                return image    }
复制代码

3. 降低峰值

通过 ARC 管理内存的对象,注册在某个 Autoreleasepool 中,Autoreleasepool 在 drain 的时候释放已经没有使用的对象。


一般没有进行特殊处理的话,会在 Runloop 结束后,有一次 Autoreleasepool 的 drain 操作,而这次 Runloop 中生成的对象也是由这个 Autoreleasepool 来管理的。这部分的原理有很多的文章介绍,这里就不多赘述了。


在图片批量处理的过程中,由于还在一个 Runloop 里,此时引用计数为 0 的对象是不会被释放的。因此需要在每次循环后触发 Autoreleasepool 的 drain 操作:


for image in images {    autoreleasepool {   operation()    }}
复制代码

4. 裁剪显示的图片

在很多场景下,图片是不会完整的显示出来的,例如下图所示的情况:



在这种情况中,即使给 UIImageView 一张完整的图片,最后渲染的时候也只会截取显示区域的 Image Buffer 去进行渲染。


这就意味着,区域外的数据,其实是没有必要的。因此在这种场景下,其实只需要裁减显示区域的图片即可。


举个例子,以前面提到 1920 * 1080 的图片为例, 显示时需要占用的内存为 829440 bytes。如果它是以 ScaleAspectFill 的方式放置在一个 300 x 300 的 UIImageView 中时,那么其实一张 300 x 300 的图片就足以展示,而此时这张图片占用的内存为 360000 bytes, 仅为前者的 43% 。


func downsample(imageAt imageURL: URL, to pointSize: CGSize, scale: CGFloat) -> UIImage {        let imageSourceOptions = [kCGImageSourceShouldCache: false] as CFDictionary        let imageSource = CGImageSourceCreateWithURL(imageURL as CFURL, imageSourceOptions)!        let maxDimensionInPixels = max(pointSize.width, pointSize.height) * scale        let downsampleOptions =            [kCGImageSourceCreateThumbnailFromImageAlways: true,             kCGImageSourceShouldCacheImmediately: true,             kCGImageSourceCreateThumbnailWithTransform: true,             kCGImageSourceThumbnailMaxPixelSize: maxDimensionInPixels] as CFDictionary        let downsampledImage =            CGImageSourceCreateThumbnailAtIndex(imageSource, 0, downsampleOptions)!        return UIImage(cgImage: downsampledImage)    }
复制代码

四、效果对比

在 App 进行优化前,是先将图片的原图显示出来,并且持有这些图片直到处理完毕。


在处理方式上,采用了 UIGraphicsBeginImageContextWithOptions 的方式来进行图片的缩放。因此造成了持续的高内存占用,峰值可以达到 600 MB 。



经过上述优化后,已经有了比较大的改观。同样的操作,总的内存占用为 221 MB,仅为之前的 36.4% 。


参考资料

[1] iOS Memory Deep Dive:


https://developer.apple.com/videos/play/wwdc2018/416


[2] Image and Graphics Best Practices:


https://developer.apple.com/videos/play/wwdc2018/219


本文转载自公众号云加社区(ID:QcloudCommunity)。


原文链接


内存占用过高怎么办?iOS图片内存优化指南


2020-09-25 14:006571

评论

发布
暂无评论
发现更多内容

终于有人把“SpringCloudAlibaba学习笔记”整理出来了

Java 程序员 架构 微服务 计算机

低代码开发平台的出现会是开发者的威胁吗

雯雯写代码

开发者 低代码 低代码开发平台

清爽视频编辑器 Tech Support

凌天一击

新一代设计编排交付套件分享

鲸品堂

交付工具

2021 年主要网络安全威胁,及时发现提早规避风险

九河云安全

百分点科技参加MIT数智未来创新峰会 入选产业数字化生态图谱

百分点科技技术团队

28天读完349页,这份阿里面试通关手册,助我闯进字节跳动

Java~~~

Java 面试 算法 多线程 架构师

【SpringCloud技术专题】「原生态Fegin」打开Fegin之RPC技术的开端,你会使用原生态的Fegin吗?(上)

码界西柚

SpringCloud OpenFegin Fegin 8月日更

FastApi-05-请求体-2

Python研究所

FastApi 8月日更

高效率程序员都在用什么工具?

狐哥说技术

效率工具

经过两年努力,我终于进入腾讯(PCG事业群4面总结)

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

什么是DPDK?DPDK的原理及学习学习路线总结

Linux服务器开发

Linux服务器开发 DPDK Linux后台开发 网络性能 网络原理

消息推送技术干货:美团实时消息推送服务的技术演进之路

JackJiang

消息推送 即时通讯 IM push

读完SpringBoot,Cloud,Nginx与Docker技术,我拿到了阿里offer

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

社招三面阿里“落榜”,幸获内推名额,4面揽下美团offer

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

987页的Java面试宝典,看完才发现,应届生求职也没那么难

Java~~~

Java 面试 微服务 多线程 架构师

二十不惑的年纪,我简直走了狗屎运(4面拿字节跳动offer)

Java~~~

Java 面试 微服务 多线程 架构师

凭借一份“面试真经pdf”,我四面字节跳动,拿下1-2级offer

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

全面到哭!BAT内部Java求职面试宝典,应届生必须人手一份

Java~~~

Java 面试 微服务 多线程 架构师

阿里巴巴大神发布的Java零基础笔记,实战教程多到手软,跪了

Java~~~

Java 面试 微服务 多线程 架构师

virtlet是什么?virtlet如何管理虚拟机?

谐云

运维安全第一步,采购堡垒机做好权限控制!

行云管家

堡垒机 安全运维 企业资产 事前授权

渣本全力以赴33天,四面阿里妈妈(淘宝联盟),拿下实习岗offer

公众号_愿天堂没有BUG

Java 编程 程序员 架构 面试

快照保护是什么意思?快照的原理是什么?

行云管家

镜像 数据保护 快照 数据安全

循序渐进带你全方位剖析原型链

加百利

大前端 原型链 自学 8月日更

网络安全界基于知识的识别和映射提出网络空间资源分类明细

郑州埃文科技

一文搞懂指标采集利器 Telegraf

尔达Erda

学习 微服务 开发者 云原生 插件开发

又一里程碑!阿里首推Java技术成长笔记,业内评级“钻石级”

Java~~~

Java redis spring 面试 架构师

如何评价Netty封装的io_uring?

BUG侦探

Netty 网络 io_uring

当新零售遇上 Serverless

Serverless Devs

阿里云 Serverless 云原生

C++20 四大特性之一:Module 特性详解

网易云信

后端

内存占用过高怎么办?iOS图片内存优化指南_大前端_云加社区_InfoQ精选文章