SDWebImage 加載顯示 WebP 與性能問題

SDWebImage 加載顯示 WebP 與性能問題

本文包含自定義下載操做 SDWebImageDownloaderOperation 與編碼器 SDWebImageCoder。SDWebImage 的版本爲 4.2.3。html

靜態圖片

對於靜態圖片來講,WebP 比 PNG 體積小,能夠省流量,可是解碼時間長。若是不須要 WebP 的原圖數據,能夠把 WebP 靜態圖片保存爲 PNG 或 JPEG,加快解碼速度。這一步能夠經過自定義下載操做 SDWebImageDownloaderOperation 實現。git

SDWebImageDownloaderOperation 的 URLSession:task:didCompleteWithError: 方法會把下載好的原圖數據 imageData 經過 callCompletionBlocksWithImage:imageData:error:finished: 方法傳給上層的回調 SDWebImageDownloaderCompletedBlock。github

能夠自定義 SDWebImageDownloaderOperation,修改 URLSession:task:didCompleteWithError: 方法,在上圖箭頭所指處修改 imageData,把靜態 WebP 圖片數據轉爲 PNG 或 JPEG 圖片數據。修改上述方法只須要添加一行代碼web

imageData = [[SDWebImageCodersManager sharedInstance] encodedDataWithImage:image format:SDImageFormatUndefined];

原圖數據有 Alpha 信息就轉爲 PNG,不然轉爲 JPEG。緩存

自定義的類是 ImageDownloaderOperation,使用這個類須要一行代碼框架

[[SDWebImageManager sharedManager].imageDownloader setOperationClass:[ImageDownloaderOperation class]];

動態圖片

WebP 格式支持動態圖片。SDWebImageWebPCoder 的 decodedImageWithData: 負責解碼,返回 UIImage。若是是動態圖片,每一幀圖片都會解碼,用全部圖片幀和總動畫時長經過 animatedImageWithImages:duration: 方法生成 UIImage。這樣作的好處是,節省 CPU 資源,不用重複解碼。對小圖片來講,比較合適。若是圖片太大,就會佔內存多。另外,若是原圖的每一幀動畫時長不相等,那麼實際播放的動畫就與原圖動畫不符。能夠用 YYImage 和 YYAnimatedImageView 來顯示動態 WebP。這樣能夠用 CPU 資源換取內存空間(YYImage 也能夠預先解碼全部圖片幀),也能夠根據原圖的每一幀動畫時長來播放動畫。直接使用 YYWebImage 框架是最方便的方法。然而,若是項目中須要統一圖片的下載、緩存管理等操做,最好只用一套圖片下載庫。這裏介紹用 SDWebImage 下載、緩存,用 YYImage 顯示 WebP 的方法。性能

YYImage 會對 WebP 進行解碼,所以不須要 SDWebImageWebPCoder 解碼全部圖片幀。自定義編碼器 SDWebImageCoder,只對第一幀圖片進行解碼,減小解碼時間。修改 decodedImageWithData: 方法,在解碼每一幀圖片的 while 循環中 (下圖箭頭所指處) 添加 break 便可,解碼成功一幀圖片就退出循環。動畫

自定義編碼器的類是 FirstFrameWebPCoder,使用這個編碼器編碼

[SDWebImageCodersManager sharedInstance].coders = @[[SDWebImageImageIOCoder sharedCoder],
                                                    [FirstFrameWebPCoder sharedCoder]];

加載顯示圖片 (cell.imageView 是 YYAnimatedImageView)url

cell.imageView.image = placeholder;
[[SDWebImageManager sharedManager] loadImageWithURL:url options:0 progress:nil completed:^(UIImage * _Nullable image, NSData * _Nullable data, NSError * _Nullable error, SDImageCacheType cacheType, BOOL finished, NSURL * _Nullable imageURL) {
    if ([image isKindOfClass:[YYImage class]]) {
        cell.imageView.image = image;
    } else if (data) {
        YYImage *yyimage = [YYImage imageWithData:data];
        cell.imageView.image = yyimage;
        NSString *key = [[SDWebImageManager sharedManager] cacheKeyForURL:url];
        [[SDWebImageManager sharedManager].imageCache storeImage:yyimage forKey:key toDisk:NO completion:nil];
    }
}];

代碼已上傳 GitHub:https://github.com/Silence-GitHub/WebPDemo

轉載請註明出處:http://www.cnblogs.com/silence-cnblogs/p/8319917.html

相關文章
相關標籤/搜索