【转】IOS高级教程1:处理1000张图片的内存优化

简介:

转载请保留以下原文链接

http://my.oschina.net/taptale/blog/91894

一、项目需求

在实际项目中,用户在上传图片时,有时会一次性上传大量的图片。在上传图片前,我们要进行一系列操作,比如:旋转图片为正确方向,压缩图片等,这些操作需要将图片加载到内存中,下面对内存的使用做详细分析.

 

二、内存分析,非优化

我在测试项目中,重复加载了一张图片1000次,首先加载图片到内存,然后进行压缩操作,释放内存

 

01 for (int i = 0; i <= 1000; i ++) {
02  
03        //1.首先我们获取到需要处理的图片资源的路径
04  
05         NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"PNG"];
06  
07         //2.将图片加载到内存中,我们使用了alloc关键字,在使用完后,可以手动快速释放掉内存
08  
09         UIImage *image = [[UIImage alloc] initWithContentsOfFile:filePath];
10  
11        //3.这一步我们将图片进行了压缩,并得到一个autorelease类型实例
12  
13         UIImage *image2 = [image imageByScalingAndCroppingForSize:CGSizeMake(480, 320)];
14  
15        //4.释放掉2步骤的内存
16  
17         [image release];
18  
19     }

上面的代码看起来没有任何问题,可以说是一种标准的代码写法,在每一步骤中都对内存做了小心的处理,我们来看一下,实际的内存使用情况:

 

 
在上图中可以看到,我们的操作在没有任何问题的情况下,在加载大量图片时,还是会造成内存的剧减

 

可以看到自动释放内存时,图片占用的内存并没有立即释放掉

 

这些资源没有立即释放的资源,占用了宝贵的内存资源,最终使程序被kill

 

三优化后的内存使用

上面程序被kill,是因为程序的内存使用问题,在上面的代码中,我们每一步都对内存做了非常小心的处理,但是在加载大量的图片时,还是会出现问题。其根本原因就是autorelease惹的祸,autorelease自动释放内存,并不会立即把内存释放掉,而是要等到下一个事件周期才会释放掉。问题是一些资源我们不得不使用autorelease类型,比如作为函数的返回值,而且系统api及项目是的大部分也都是这么做的,如果全都依靠我们手动释放很容易造成内存泄漏。

 

记住:NSAutoreleasePool里面的维护了一个NSMutableArray数组,所有标记为autorelease的对象都会被添加都该数组中。只有当pool对象被drain的时候,才会去遍历该数组,若retainCount为0则释放内存,不为零就发生内存泄露!OC已经为我们建立一个pool对象,但是该pool对象需要比较久的时间才能drain掉,因此在一些遍历处理的场景中,需要我们手动去建立pool对象,并手动drain掉。

 

01 for (int i = 0; i <= 1000; i ++) {
02  
03        //创建一个自动释放池
04  
05         NSAutoreleasePool *pool = [NSAutoreleasePool new];
06  
07         NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"PNG"];
08  
09         UIImage *image = [[UIImage alloc] initWithContentsOfFile:filePath];
10  
11         UIImage *image2 = [image imageByScalingAndCroppingForSize:CGSizeMake(480, 320)];
12  
13         [image release];
14  
15        //将自动释放池内存释放,它会同时释放掉上面代码中产生的临时变量image2
16  
17         [pool drain];
18  
19     }

 

优化后的,内存使用情况

 

 可用内存不再明显的减少

 

CGImage及UIImage的数据由原来的220多减少到6-7个

 

可以看到使用了 NSAutoreleasePool后,加载大量图片的时候内存也不会出现问题

四、自动释放池概述

(1)自动释放池被置于一个堆栈中,虽然它们通常被称为被“嵌套”的。当您创建一个新的自动释放池时,它被添加到堆栈的顶部。当自动释放池被回收时,它们从堆栈中被删除。当一个对象收到送autorelease消息时,它被添加到当前线程的目前处于栈顶的自动释放池中。你不能向自动释放池发送autorelease或retain消息Application Kit会在一个事件周期(或事件循环迭代)的开端—比如鼠标按下事件—自动创建一个自动释放池,并且在事件周期的结尾释放它,因此您的代码通常不必关心。 有三种情况您应该使用您自己的自动释放池:

  • 如果您正在编写一个不是基于Application Kit的程序,比如命令行工具,则没有对自动释放池的内置支持;您必须自己创建它们。

  • 如果您生成了一个从属线程,则一旦该线程开始执行,您必须立即创建您自己的自动释放池;否则,您将会泄漏对象。

  •  如果您编写了一个循环,其中创建了许多临时对象,您可以在循环内部创建一个自动释放池,以便在下次迭代之前销毁这些
     对象。这可以帮助减少应用程序的最大内存占用量。

(2) release和drain之间的差异

      在引用计数环境下,release和drain一样,会直接自动释放池l对象。

      在GC(垃圾回收)环境下,release是一个no-op(空操作),drain会触发垃圾回收(如果自上次垃圾回收以来分配的内存大于当前的阈值)。

      通常情况下,您都应该使用drain而不是使用release来销毁自动释放池。

     -drain方法只适用于Mac OS X10.4(Tiger)及更高版本。

     在OS X Mountain Lion v10.8操作系统下,GC(垃圾回收)将被废弃,ARC(Automatic Reference Counting自动引用计数)为推荐的替代技术。


本文转自编程小翁博客园博客,原文链接:http://www.cnblogs.com/wengzilin/p/3301549.html,如需转载请自行联系原作者

相关文章
|
6月前
|
JSON JavaScript 安全
iOS应用程序数据保护:如何保护iOS应用程序中的图片、资源和敏感数据
iOS应用程序数据保护:如何保护iOS应用程序中的图片、资源和敏感数据
58 1
|
6月前
|
Shell Linux C语言
【Shell 命令集合 磁盘维护 】Linux 创建一个初始化内存盘 mkinitrd命令使用教程
【Shell 命令集合 磁盘维护 】Linux 创建一个初始化内存盘 mkinitrd命令使用教程
97 0
|
6月前
|
存储 缓存 安全
基于iOS平台的高效图片缓存策略实现
【4月更文挑战第22天】 在移动应用开发中,图片资源的加载与缓存是影响用户体验的重要因素之一。尤其对于iOS平台,由于设备存储空间的限制以及用户对流畅性的高要求,设计一种合理的图片缓存策略显得尤为关键。本文将探讨在iOS环境下,如何通过使用先进的图片缓存技术,包括内存缓存、磁盘缓存以及网络请求的优化,来提高应用的性能和响应速度。我们将重点分析多级缓存机制的设计与实现,并对可能出现的问题及其解决方案进行讨论。
|
14天前
|
存储 前端开发 Java
Kotlin教程笔记 - MVVM架构怎样避免内存泄漏
Kotlin教程笔记 - MVVM架构怎样避免内存泄漏
|
17天前
|
存储 前端开发 Java
Kotlin教程笔记 - MVVM架构怎样避免内存泄漏
Kotlin教程笔记 - MVVM架构怎样避免内存泄漏
33 0
|
6月前
|
存储 缓存 算法
实现iOS平台的高效图片缓存策略
【4月更文挑战第22天】在移动应用开发中,图片资源的处理是影响用户体验的重要因素之一。特别是对于图像资源密集型的iOS应用,如何有效地缓存图片以减少内存占用和提升加载速度,是开发者们面临的关键挑战。本文将探讨一种针对iOS平台的图片缓存策略,该策略通过结合内存缓存与磁盘缓存的机制,并采用先进的图片解码和异步加载技术,旨在实现快速加载的同时,保持应用的内存效率。
|
3月前
|
测试技术 开发工具 虚拟化
iOS自动化测试方案(一):MacOS虚拟机保姆级安装Xcode教程
这篇文章提供了一份保姆级的教程,指导如何在MacOS虚拟机上安装Xcode,包括环境准备、基础软件安装以及USB扩展插件的使用,以实现iOS自动化测试方案的第一步。
72 0
iOS自动化测试方案(一):MacOS虚拟机保姆级安装Xcode教程
|
6月前
|
存储 缓存 编解码
实现iOS平台的高效图片缓存策略
【4月更文挑战第23天】在移动应用开发领域,尤其是图像处理密集型的iOS应用中,高效的图片缓存策略对于提升用户体验和节省系统资源至关重要。本文将探讨一种针对iOS平台设计的图片缓存方案,该方案通过结合内存缓存与磁盘缓存的多层次结构,旨在优化图片加载性能并降低内存占用。我们将深入分析其设计理念、核心组件以及在实际场景中的应用效果,同时对比其他常见缓存技术的优势与局限。
|
3月前
|
Swift iOS开发
iOS开发-属性的内存管理
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),不维持对象生命期,可用于解除循环引用;`assign`(赋值),适用于基本数据类型及非指针对象属性;`copy`,复制对象而非引用,确保对象不变性。iOS采用引用计数管理内存,ARC(自动引用计数)自动处理引用增减,简化开发。为避免循环引用,可利用弱引用或Swift中的`[weak self]`。最佳实践包括:选择恰当的内存管理策略、减少不必要的强引用、及时释放不再使用的对象、注意block内存管理,并使用Xcode工具进行内存分析。
|
3月前
|
图形学 数据安全/隐私保护 iOS开发
Unity与IOS⭐Xcode打包,上架TestFlight的完整教程
Unity与IOS⭐Xcode打包,上架TestFlight的完整教程