最近遇到一個(gè)保存圖片時(shí)的崩潰售睹, 崩潰信息如下:
0 CoreFoundation!__exceptionPreprocess + 0x84
1 libobjc.A.dylib!objc_exception_throw + 0x38
2 CoreFoundation!+[NSException raise:format:] + 0x7c
3 Foundation!_NSMutableDataGrowBytes + 0x300
4 Foundation!-[NSConcreteMutableData appendBytes:length:] + 0x174
5 ImageIO!CGImageWriteSessionPutBytes + 0x94
6 ImageIO!write_fn + 0x28
7 ImageIO!png_write_chunk_data + 0x34
8 ImageIO!_cg_png_write_complete_chunk + 0x40
9 ImageIO!png_compress_IDAT + 0x160
10 ImageIO!png_write_find_filter + 0xa80
11 ImageIO!_cg_png_write_row + 0x300
12 ImageIO!writeOnePng + 0x1358
13 ImageIO!_CGImagePluginWritePNG + 0x90
14 ImageIO!CGImageDestinationFinalize + 0x1e8
15 UIKit!UIImagePNGRepresentation + 0x248
16 MFAFImageCache addImage:forRequest:withAdditionalIdentifier:
提示的異常信息是:
Exception Name: NSMallocException
Exception Reason: *** -[NSConcreteMutableData appendBytes:length:]: unable to allocate memory for length (1751849)
查看一下調(diào)用函數(shù), 主要調(diào)用如下:
NSString *filePath = [FileModel getLocalFileName:kPhotosDirectory byUrl:[[request URL] absoluteString]];
if (image) {
@try {
NSData *imageData = UIImagePNGRepresentation(image);
[imageData writeToFile:filePath atomically:YES];
} @catch (NSException *exception) {
MFLogError(@"ImageCache", @"save image error:%@", exception);
} @finally {
}
}
說明在執(zhí)行 NSData *imageData = UIImagePNGRepresentation(image)
這個(gè)操作時(shí)泣崩,崩潰了壳炎。而且崩潰的原因是內(nèi)存分配出錯(cuò)蔓榄。 通常這種問題都是因?yàn)閮?nèi)存不足引起的。但是查看了一下log峭弟, 發(fā)現(xiàn)并沒有 MemoryWarning 這句日志朗伶,說明程序還沒來得及處理系統(tǒng)的內(nèi)存警告就崩潰了。
根據(jù)這些現(xiàn)象猜測可能的原因
- 圖片比較大
- 外層也許有個(gè)for循環(huán)昨稼,多次調(diào)用节视,導(dǎo)致臨時(shí)對(duì)象沒來得及馬上釋放
針對(duì)這兩種猜測,作了相應(yīng)的處理
圖片內(nèi)容較大
其實(shí)有 UIImage 對(duì)象的時(shí)候假栓,說明圖片已經(jīng)可以正常展示寻行,加載這張圖片到內(nèi)存暫時(shí)是沒有問題的。但保存的時(shí)候又根據(jù) UIImage 對(duì)象生成 NSData但指, 然后在通過NSData保存文件寡痰。 在NSData寫入文件到釋放前,內(nèi)存里面其實(shí)有兩份 圖片 的數(shù)據(jù)棋凳, 一份是顯示的拦坠, 一份是準(zhǔn)備用來保存的NSData。 這樣其實(shí)就是臨時(shí)分配了一塊大空間剩岳,如果可以減少臨時(shí)對(duì)象的分配贞滨,直接讀取UIImage的數(shù)據(jù)來存儲(chǔ)就可以降低內(nèi)存的使用。
通過查閱文檔,發(fā)現(xiàn)Image IO 的庫可以做到晓铆。
于是勺良,保存函數(shù)改成這樣:
+ (BOOL)saveImage:(UIImage *)image toFile:(NSString *)filePath
{
if (!image.CGImage) {
return NO;
}
CFURLRef url = (__bridge CFURLRef)[NSURL fileURLWithPath:filePath];
CGImageDestinationRef destination = CGImageDestinationCreateWithURL(url, kUTTypePNG, 1, nil);
if (!destination) {
return NO;
}
CGImageDestinationAddImage(destination, image.CGImage, nil);
CGImageDestinationFinalize(destination);
CFRelease(destination);
return YES;
}
臨時(shí)對(duì)象沒有馬上釋放
這種情況比較好處理,就是因?yàn)榧拥?AutoReleasePool 的對(duì)象沒有馬上釋放骄噪,必須要等下一次 RunLoop 循環(huán)才會(huì)清理尚困。
這種問題可以加 @autoreleasepool{} 處理。
最后函數(shù)變成:
+ (BOOL)saveImage:(UIImage *)image toFile:(NSString *)filePath
{
if (!image.CGImage) {
return NO;
}
@autoreleasepool {
CFURLRef url = (__bridge CFURLRef)[NSURL fileURLWithPath:filePath];
CGImageDestinationRef destination = CGImageDestinationCreateWithURL(url, kUTTypePNG, 1, nil);
if (!destination) {
return NO;
}
CGImageDestinationAddImage(destination, image.CGImage, nil);
CGImageDestinationFinalize(destination);
CFRelease(destination);
}
return YES;
}
當(dāng)然 @autoreleasepool 寫在這里作用并不大链蕊, 應(yīng)該要加在相應(yīng)的for()循環(huán)里面事甜。
因此定義一個(gè)宏,方便替換滔韵,只要把原來的 for 替換成 AutoRelease_for 就可以逻谦。
#define AutoRelease_for(...) for(__VA_ARGS__) @autoreleasepool