由于已知原因的内存压力终止,未知的解决方案

时间:2013-10-31 15:54:27

标签: ios objective-c memory

由于应用程序使用中某个点的内存压力,我的应用程序正在终止,我已将问题分解为造成问题的一大块代码。

我将复制下面的代码块,但首先我将描述它正在做什么。

说明

我有一个迭代浏览视频列表的for循环。对于每个视频,for循环增加包含滚动视图的大小,绘制标签和按钮(均与视频有关),并异步抓取视频的缩略图并将其放在按钮下方。

问题

抓取缩略图部分是问题所在。我不认为这是异步完成的事实是问题,因为我已经同步尝试并且终止仍然发生。当我注释掉抓取缩略图的代码(下面代码中的整个异步部分)时,应用程序不会崩溃。

守则

注意:为简洁起见,我在某些情况下使用注释替换代码。

for (int i = [_videoURLs count]-1; i >= 0 ; i--)
    {
        //increase the scrollview size

        //get background image:
        dispatch_async(screenshotQueue, ^{

            AVURLAsset *as = [[AVURLAsset alloc] initWithURL:currentURL options:nil];
            AVAssetImageGenerator *ima = [[AVAssetImageGenerator alloc] initWithAsset:as];
            ima.appliesPreferredTrackTransform = YES;
            NSError *err = NULL;
            CMTime time = CMTimeMake(1, 24);
            CGImageRef imgRef = [ima copyCGImageAtTime:time actualTime:NULL error:&err];
            UIImage *thumbnail = [[UIImage alloc] initWithCGImage:imgRef];

            dispatch_async(dispatch_get_main_queue(), ^{

                UIImageView *backgroundImage = [[UIImageView alloc]initWithFrame:buttonFrame];
                backgroundImage.image = thumbnail;
                backgroundImage.layer.opacity = 0;
                [self.videoScrollView addSubview:backgroundImage];

                [self.videoScrollView sendSubviewToBack:backgroundImage];

                [UIView animateWithDuration:0.5 delay:0.0 options: UIViewAnimationOptionCurveEaseInOut
                 animations:^{
                     backgroundImage.layer.opacity = 1;
                 }
                 completion:^(BOOL finished){}];

            });

        });

        //add title to the screen caps

        //draw the button

    }

我正在使用Xcode 5,在iOS 7的新设备上进行测试。我是自学成才,所以我确信我已经找到了一些不良习惯,但我希望我能够错过了一些明显有经验的人会接受的东西。

我已经尝试了大量的谷歌搜索并搜索堆栈溢出,并尝试查看诊断,日志和内存配置文件以进一步隔离问题,但无济于事。

非常感谢任何帮助!

修改

如何重现错误:

  1. 转到视频列表
  2. 保留视频列表
  3. 重复1 + 2 4或5次导致崩溃
  4. 步骤2.简单地从其超级视图中删除整个scrollView,然后重绘它。我知道每次回到视频列表都没有必要重新绘制视频列表,但我出于某种原因这样做,我不相信这是问题的根源(我可能是错的)。

    编辑2

    这是内存配置文件的图像: enter image description here

2 个答案:

答案 0 :(得分:12)

您展示的代码存在巨大泄漏。您正在创建一个Core Foundation对象:

CGImageRef imgRef = [ima copyCGImageAtTime:time actualTime:NULL error:&err];

但我看不到相应的CFRelease

你的代码应该是:

CGImageRef imgRef = [ima copyCGImageAtTime:time actualTime:NULL error:&err];
UIImage *thumbnail = [[UIImage alloc] initWithCGImage:imgRef];
CFRelease(imgRef);

答案 1 :(得分:2)

好吧,所以我找到了一个不错的解决方案。我仍然不完全确定为什么这首先是一个问题,但我有一些怀疑。

我认为生成缩略图的过程太过密集,无法在for循环上重复执行。我有这种预感,所以决定调整上面的代码。我不是每次都生成一个新的缩略图,而是将新生成的屏幕截图保存为文件主目录的png。然后下次,我会从那里加载这个资源。这导致内存问题消失。

我希望这可以帮助有类似问题的人 - 如果有人有任何需要补充或者有更好的答案,我会全力以赴。