ios系统 处理内存警告

iPhone下每个app可用的内存是被限制的,如果一个app使用的内存超过20M,则系统会向该app发送Memory Warning消息。收到此消息后,app必须正确处理,否则可能出错或者出现内存泄露。

app收到Memory Warning后会调用:UIApplication::didReceiveMemoryWarning -> UIApplicationDelegate::applicationDidReceiveMemoryWarning,然后调用当前所有的viewController进行处理。因此处理的主要工作是在viewController。

当我们的程序在第一次收到内存不足警告时,应该释放一些不用的资源,以节省部分内存。否则,当内存不足情形依然存在,iOS再次向我们程序发出内存不足的警告时,我们的程序将会被iOS kill掉。

iOS的UIViewController 类给我们提供了处理内存不足的接口。在iOS 3.0 之前,当系统的内存不足时,UIViewController的didReceiveMemoryWarining 方法会被调用,我们可以在didReceiveMemoryWarining 方法里释放掉部分暂时不用的资源。

从iOS3.0 开始,UIViewController增加了viewDidUnload方法。该方法和viewDIdLoad相配对。当系统内存不足时,首先UIViewController的didReceiveMemoryWarining 方法会被调用,而didReceiveMemoryWarining 会判断当前ViewController的view是否显示在window上,如果没有显示在window上,则didReceiveMemoryWarining 会自动将viewcontroller 的view以及其所有子view全部销毁,然后调用viewcontroller的viewdidunload方法。如果当前UIViewController的view显示在window上,则不销毁该viewcontroller的view,当然,viewDidunload也不会被调用了。但是到了ios6.0之后,这里又有所变化,ios6.0内存警告的viewDidUnload 被屏蔽,即又回到了ios3.0的时期的内存管理方式。

iOS3-iOS5.0以前版本收到内存警告: 调用didReceiveMemoryWarning内调用super的didReceiveMemoryWarning会将controller的view进行释放。所以我们不能将controller的view再次释放。 处理方法:

Java代码

  1. -(void)didReceiveMemoryWarning
  2. {
  3. [super didReceiveMemoryWarning];//如没有显示在window上,会自动将self.view释放。
  4. // ios6.0以前,不用在此做处理,self.view释放之后,会调用下面的viewDidUnload函数,在viewDidUnload函数中做处理就可以了。
  5. }
  6. -(void)viewDidUnload
  7. {
  8. // Release any retained subviews of the main view.不包含self.view
  9. //处理一些内存和资源问题。
  10. [super viewDidUnload];
  11. }
 -(void)didReceiveMemoryWarning
        {
                 [super didReceiveMemoryWarning];//如没有显示在window上,会自动将self.view释放。
                 // ios6.0以前,不用在此做处理,self.view释放之后,会调用下面的viewDidUnload函数,在viewDidUnload函数中做处理就可以了。
        }
        -(void)viewDidUnload
        {
               // Release any retained subviews of the main view.不包含self.view
               //处理一些内存和资源问题。
                [super viewDidUnload];

        }

iOS6.0及以上版本的内存警告: 调用didReceiveMemoryWarning内调用super的didReceiveMemoryWarning调只是释放controller的resouse,不会释放view 处理方法:     -(void)didReceiveMemoryWarning     {             [super didReceiveMemoryWarning];//即使没有显示在window上,也不会自动的将self.view释放。             // Add code to clean up any of your own resources that are no longer necessary.
            // 此处做兼容处理需要加上ios6.0的宏开关,保证是在6.0下使用的,6.0以前屏蔽以下代码,否则会在下面使用self.view时自动加载viewDidUnLoad

if ([[UIDevice currentDevice].systemVersion floatValue] >= 6.0) {

//需要注意的是self.isViewLoaded是必不可少的,其他方式访问视图会导致它加载 ,在WWDC视频也忽视这一点。

if (self.isViewLoaded && !self.view.window)// 是否是正在使用的视图              {                    // Add code to preserve data stored in the views that might be                    // needed later.                             // Add code to clean up other strong references to the view in                    // the view hierarchy.                    self.view = nil;// 目的是再次进入时能够重新加载调用viewDidLoad函数。              }

}     }

但是似乎这么写相对于以前并不省事。最终我们找到一篇文章,文章中说其实并不值得回收这部分的内存,原因如下:

1. UIView是UIResponder的子类,而UIResponder有一个CALayer的成员变量,CALayer是具体用于将自己画到屏幕上的。

2. CALayer是一个bitmap图象的包装类,当UIView调用自身的drawRect时,CALayer才会创建这个bitmap图象类。

3. 具体占内存的其实是一个bitmap图象类,CALayer只占48bytes, UIView只占96bytes。而一个iPad的全屏UIView的bitmap类会占到12M的大小!

4.在iOS6时,当系统发出MemoryWarning时,系统会自动回收bitmap类。但是不回收UIView和CALayer类。这样即回收了大部分内存,又能在需要bitmap类时,根据CALayer类重建。

所以,iOS6这么做的意思是:我们根本没有必要为了几十byte而费力回收内存。

--------------------------切糕分割线--------------

PS:

1、关于这个的官方文档:https://developer.apple.com/library/ios/#featuredarticles/ViewControllerPGforiPhoneOS/ViewLoadingandUnloading/ViewLoadingandUnloading.html

2、zon2012貌似都没有ios6的这个兼容(其实view是没问题的,关键是资源)

时间: 2024-10-26 08:24:30

ios系统 处理内存警告的相关文章

iOS当发生内存警告时的一些处理

//释放暂时不使用的内存,共当前程序使用. - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; // Dispose of any resources that can be recreated. //当收到内存警告时,移除当前没有在屏幕上显示的视图. //判断是否可以安全的移除视图控制器的view. //判断当前的视图控制器的view是否是屏幕上显示.self.view.window //判断当前视图控制器的vi

ios6前后分别如何处理内存警告

这里有一篇文章,很详细地说明了ios6前后是如何处理内存警告的: 来自唐巧的技术博客:http://blog.devtang.com/blog/2013/05/18/goodbye-viewdidunload/ 1.ios4和ios5中,当收到内存警告时,系统会自动调用当前没有在界面上的VC的viewDidUnload方法. 在viewDidUnload写下如下代码: if([self isViewLoaded] && ![[self view] window]) { [self setV

IOS中的内存不足警告处理(译)

由于在IOS中虚拟内存系统不会采用页置换的方式来获取请求内存,取而代之的是它通过移除应用程序中的强引用来释放一些内存资源,我们知道强引用在IOS中表示拥有关系,只要有至少一个变量拥有这个对象,那么对象就不会被释放,当然IOS中没有类似Java或.NET的垃圾回收机制,所以我们需要手动来释放对象的引用.当然在ARC中,编译器会在编译阶段为我们插入这段释放代码.释放其实就是减少引用计数,有关于引用计数在内存管理中的应用,我会在另一篇文章中做描述. 当可用虚拟内存页的数量低于系统规定的临界值时,系统会

为什么安卓系统大内存却比IOS卡顿

内存管理,深受Windows下释放内存加速系统毒害,人们误以为内存剩余越多越好.其实不然,当一个应用程序退出时,OS不应该立即释放其内存,而是要等到内存不够用了才开始释放.Why? (1)有些应用频繁启动关闭,启动时,如果程序已经加载到了内存中,那么启动速度将飞快.相反的,如果不在内存,就需要从外部存储里面读取,需要等待,比较慢. (2)内存使用率越高越好.内存是易失性存储设备,当断电了数据就会消失,使用时还需要不断刷新.刷新0,还是1,对于功率是不会产生大的影响,所以也不会存在,内存使用率低,

iOS内存警告处理

- (void)didReceiveMemoryWarning//内存警告{    [super didReceiveMemoryWarning];    // Dispose of any resources that can be recreated.        NSLog(@"内存快爆了"); //判断是否加载过 或者是否正在屏幕上显示    if([self isViewLoaded]==YES &&self.view.window==nil){      

iOS内存警告浅析

在开发中,内存的资源是很宝贵的.所以合理的管理项目的内存是恨重要的. 在项目中,最消耗内存的,莫过于图片.我们都知道,最常用的获取图片的方式是[UIImage imageName:@"image"];是这种方式,这种方式有一个机制,就是自动的有缓存.还有一种常用的方式是:[UIImage imageWithContentsOfFile:"fileName"];这种方式是没有缓存的.总结:由上可知,如果你想获娶大的图片,再加上不常用到此图片,建议使用第二种方式来加载图

iOS性能优化/内存优化常用方法

1. 用ARC管理内存 ARC(Automatic ReferenceCounting, 自动引用计数)和iOS5一起发布,它避免了最常见的也就是经常是由于我们忘记释放内存所造成的内存泄露.它自动为你管理retain和release的过程,所以你就不必去手动干预了.忘掉代码段结尾的release简直像记得吃饭一样简单.而ARC会自动在底层为你做这些工作.除了帮你避免内存泄露,ARC还可以帮你提高性能,它能保证释放掉不再需要的对象的内存. 2. 在正确的地方使用 reuseIdentifier 一

解读AppIcon图标设置置信息和App内存警告临界值

前面有同学问到了iOS内存警告临界值和工程项目里AppIcon的一些配置信息,相信对刚入行的同学来说,可能都会碰到类似的问题,记录一下供后来者查询. 1.先简单说下AppIcon的图标的配置信息 1).在右边的App Icon勾选项,暂时只选择了支持iPhone iOS7.0 and Later, 表示这些icon只支持iPhone设备(包括iPod)的iOS系统最低7.0至最新系统: 2).2x图标表示支持在Retina显示屏上,3x图标表示支持在Plus屏幕尺寸5.5英寸上显示: 3).再看

iOS MRC ARC 内存管理

转自:http://www.jianshu.com/p/48665652e4e4 1. 什么是内存管理 程序在运行的过程中通常通过以下行为,来增加程序的的内存占用 创建一个OC对象 定义一个变量 调用一个函数或者方法 而一个移动设备的内存是有限的,每个软件所能占用的内存也是有限的 当程序所占用的内存较多时,系统就会发出内存警告,这时就得回收一些不需要再使用的内存空间.比如回收一些不需要使用的对象.变量等 如果程序占用内存过大,系统可能会强制关闭程序,造成程序崩溃.闪退现象,影响用户体验 所以,我