iOS[UIImage imageNamed:]生成的对象释放时机

如果使用imageNamed这个方法加载一些比较大的资源文件就容易崩溃,从而引发了imageNamed生成的对象什么时候释放?

使用imageNamed这个方法生成的UIImage对象,会在应用的bundle中寻找图片,如果找到,则Cache到系统缓存中,作为内存的cache,而程序员是无法操作cache的,只能由系统自动处理,如果我们需要重复加载一张图片,那这无疑是一种很好的方式,因为系统能很快的从内存的cache找到这张图片,但是试想,如果加载很多很大的图片的时候,内存消耗过大的时候,就会会强制释放内存,即会遇到内存警告(memory warnings)。由此看来[UIImage imageNamed:]只适合与UI界面中小的贴图的读取,而一些比较大的资源文件应该尽量避免使用这个接口[UIImage imageWithContentsOfFile]解决掉这个问题。

[[UIImageView alloc] init]还有一些其他的 init 方法,返回的都是 autorelease 对象。而 autorelease 不能保证什么时候释放,所以不一定在引用计数为 0 就立即释放,只能保证在 autoreleasepool 结尾的时候释放。

for (int i = 0; i < 1000; i++) {
    UIImage* image = [UIImage imageNamed:@"some_image"];
    // 对 image 进行一些处理,比如存文件什么的
}

执行这段代码就会看到内存越增越大,容易导致崩溃。而在每一次循环结束的时候,UIImage 引用都为0了,不过系统不会把它立即释放掉;循环次数多了内存就爆掉了。

为了解决这个问题,可以改成这样:

for (int i = 0; i < 1000; i++) {
    @autoreleasepool {
        UIImage* image = [UIImage imageNamed:@"some_image"];
        // 对 image 进行一些处理,比如存文件什么的
    }
}

这样在每次循环结束的时候都会立即释放 UIImage,也不会对内存造成压力了。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,142评论 1 32
  • 1. 用ARC管理内存 ARC(Automatic ReferenceCounting, 自动引用计数)和iOS5...
    Icec阅读 1,567评论 0 7
  • 当我们开发iOS应用时,好的性能对我们的App来说是很重要的。你的用户也希望如此,但是如果你的app表现的反应迟钝...
    iOS开发攻城狮阅读 1,549评论 0 14
  • 面向对象的三大特性:封装、继承、多态 OC内存管理 _strong 引用计数器来控制对象的生命周期。 _weak...
    运气不够技术凑阅读 1,135评论 0 10
  • 呼啸而至,一幅落日美景图就这样不期而遇地出现在我的眼前。这可能是成千上万个日子里最普通的一天,山痕、落日、云朵以及...
    玛丽莲不是我阅读 568评论 0 0