本文包含自定義下載操做 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