5

我正在处理的应用程序图像。这就像用户最多放置 4 张图像,并根据用户选择的模板对它们进行应用布局。在最终视图中可能会添加 2-3 次图像。

布局中的每个图像都在 NSView 中绘制(使用 drawInRect 方法的 NSView 的 drawRect 方法)。现在通过将 NSView 保存为 Image 创建最终图像(通过布局所有图像来组合图像),并且一切正常。

现在我面临的问题是,一旦所有处理完成,应用程序就会保留内存。我使用了仪器分配,我没有看到内存泄漏,但我看到“持久字节”随着应用程序的每个会话而不断增加,并且一个用户报告了 GB 的问题。请看截图。

仪器内存使用情况 图像IO_TIFF 当我在 Instruments 中进一步调查时,我在下面看到导致内存保留的应用程序的代码快照。都与 ImageIO 和 coreImages 有关。从仪器看下面:

仪器代码快照

然而,这似乎只是 10.10 及以上系统的问题。在 10.9.x 中测试了相同版本的应用程序,内存使用量保持在 60MB。在应用程序中的会话执行期间,它会达到 200MB,但一旦完成,它就会恢复到通常用于某种应用程序的 50-60MB。

[_photoImage drawInRect: self.bounds fromRect: NSZeroRect operation: NSCompositeSourceOver fraction: 1.0 respectFlipped: YES hints: nil];
            _photoImage = nil;

以上代码我用于在 NSView 的 drawRect 方法中绘制图像,图像中显示的代码用于将 NSView 作为图像。

更新:经过进一步调查,我发现缓存 NSI​​mage 的 TIFF 数据的是CGImageSourceCreateWithData。我更多地使用下面的代码来裁剪图像,如果我取消注释它的内存消耗就可以正常工作。

NSData *imgData = [imageToCrop TIFFRepresentation];
CGImageSourceRef source = CGImageSourceCreateWithData((CFDataRef)imgData, NULL);
CGImageRef maskRef =  CGImageSourceCreateImageAtIndex(source, 0, NULL);
CGImageRef imageRef = CGImageCreateWithImageInRect(maskRef, rect);
NSImage *cropped = [[NSImage alloc] initWithCGImage: imageRef size:rect.size];

CGImageRelease(maskRef);
CGImageRelease(imageRef);
CFRelease(source);
//CFRelease( options );
imgData = nil;

我还尝试将kCGImageSourceShouldCache 显式设置为 false(但默认情况下为 false),但结果相同。请帮助解决内存保留问题。

4

1 回答 1

4

最后经过大量调试,结果发现CGImageSourceCreateWithData在某处保留了 NSImage 的 TIFF 数据。当我改变这一行时:

CGImageSourceRef source = CGImageSourceCreateWithData((CFDataRef)imgData, NULL);

CGImageSourceRef source = CGImageSourceCreateWithURL((CFURLRef)[NSURL fileURLWithPath:path], NULL);

一切刚刚开始工作正常,应用程序的内存使用量从 300MB(6 张图像)下降到 50-60MB,现在它的行为一致。

除了上述更改之外,它仍然会在某处导致内存保留,因此要摆脱它,在所有处理完成后,我将每一层的图像清除为“nil”,这就像魅力一样。我的印象是,将 parent 设为“nil”也会发布图像,但这不起作用。

无论如何,如果有人似乎对 drawInRect 或 cacheDisplayInRect 有问题,那么如果以后不需要,请确保清除图像。

2016 年 7 月 2 日更新

我发现kCGImageSourceShouldCache在 32 位中默认为 false,在 64 位中默认为 true。通过将其设置为 false,我能够使用以下代码释放内存。

const void *keys[] =   { kCGImageSourceShouldCache};
    const void *values[] = { kCFBooleanFalse};
    CFDictionaryRef optionsDictionary = CFDictionaryCreate(NULL, keys, values, 1, NULL, NULL);

    CGImageSourceRef source = CGImageSourceCreateWithData((__bridge CFDataRef)[image TIFFRepresentation], optionsDictionary);

希望它可以帮助某人。

于 2016-03-04T12:20:19.357 回答