亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS中圖片的解壓縮到渲染過程詳解

2020-07-26 02:20:59
字體:
來源:轉載
供稿:網友

前言

在移動app開發過程中,圖片往往是不可或缺的資源。從磁盤上加載一張圖片,到顯示到屏幕上,中間經過了一些復雜的過程,其中非常重要的一步就是對圖片的解壓縮。下面來一起看看詳細的介紹吧

一.圖像從文件到屏幕過程

通常計算機在顯示是CPU與GPU協同合作完成一次渲染.接下來我們了解一下CPU/GPU等在這樣一次渲染過程中,具體的分工是什么?

  • CPU: 計算視圖frame,圖片解碼,需要繪制紋理圖片通過數據總線交給GPU
  • GPU: 紋理混合,頂點變換與計算,像素點的填充計算,渲染到幀緩沖區。
  • 時鐘信號:垂直同步信號V-Sync / 水平同步信號H-Sync。
  • iOS設備雙緩沖機制:顯示系統通常會引入兩個幀緩沖區,雙緩沖機制

圖片顯示到屏幕上是CPU與GPU的協作完成

對應應用來說,圖片是最占用手機內存的資源,將一張圖片從磁盤中加載出來,并最終顯示到屏幕上,中間其實經過了一系列復雜的處理過程。

二.圖片加載的工作流程

1、假設我們使用 +imageWithContentsOfFile: 方法從磁盤中加載一張圖片,這個時候的圖片并沒有解壓縮;

2、然后將生成的 UIImage 賦值給 UIImageView ;

3、接著一個隱式的 CATransaction 捕獲到了 UIImageView 圖層樹的變化;

4、在主線程的下一個 runloop 到來時,Core Animation 提交了這個隱式的 transaction ,這個過程可能會對圖片進行 copy 操作,而受圖片是否字節對齊等因素的影響,這個 copy 操作可能會涉及以下部分或全部步驟:

  • 分配內存緩沖區用于管理文件 IO 和解壓縮操作;
  • 將文件數據從磁盤讀到內存中;
  • 將壓縮的圖片數據解碼成未壓縮的位圖形式,這是一個非常耗時的 CPU 操作;
  • 最后 Core Animation 中CALayer使用未壓縮的位圖數據渲染 UIImageView 的圖層。
  • CPU計算好圖片的Frame,對圖片解壓之后.就會交給GPU來做圖片渲染

5、渲染流程

  • GPU獲取獲取圖片的坐標
  • 將坐標交給頂點著色器(頂點計算)
  • 將圖片光柵化(獲取圖片對應屏幕上的像素點)
  • 片元著色器計算(計算每個像素點的最終顯示的顏色值)
  • 從幀緩存區中渲染到屏幕上

我們提到了圖片的解壓縮是一個非常耗時的 CPU 操作,并且它默認是在主線程中執行的。那么當需要加載的圖片比較多時,就會對我們應用的響應性造成嚴重的影響,尤其是在快速滑動的列表上,這個問題會表現得更加突出。

三.為什么要解壓縮圖片

既然圖片的解壓縮需要消耗大量的 CPU 時間,那么我們為什么還要對圖片進行解壓縮呢?是否可以不經過解壓縮,而直接將圖片顯示到屏幕上呢?答案是否定的。要想弄明白這個問題,我們首先需要知道什么是位圖

其實,位圖就是一個像素數組,數組中的每個像素就代表著圖片中的一個點。我們在應用中經常用到的 JPEG 和 PNG 圖片就是位圖

大家可以嘗試

UIImage *image = [UIImage imageNamed:@"text.png"];CFDataRef rawData = CGDataProviderCopyData(CGImageGetDataProvider(image.CGImage));

打印rawData,這里就是圖片的原始數據.

事實上,不管是 JPEG 還是 PNG 圖片,都是一種壓縮的位圖圖形格式。只不過 PNG 圖片是無損壓縮,并且支持 alpha 通道,而 JPEG 圖片則是有損壓縮,可以指定 0-100% 的壓縮比。值得一提的是,在蘋果的 SDK 中專門提供了兩個函數用來生成 PNG 和 JPEG 圖片:

// return image as PNG. May return nil if image has no CGImageRef or invalid bitmap formatUIKIT_EXTERN NSData * __nullable UIImagePNGRepresentation(UIImage * __nonnull image);// return image as JPEG. May return nil if image has no CGImageRef or invalid bitmap format. compression is 0(most)..1(least)       UIKIT_EXTERN NSData * __nullable UIImageJPEGRepresentation(UIImage * __nonnull image, CGFloat compressionQuality);

因此,在將磁盤中的圖片渲染到屏幕之前,必須先要得到圖片的原始像素數據,才能執行后續的繪制操作,這就是為什么需要對圖片解壓縮的原因。

四.解壓縮原理

既然圖片的解壓縮不可避免,而我們也不想讓它在主線程執行,影響我們應用的響應性,那么是否有比較好的解決方案呢?

我們前面已經提到了,當未解壓縮的圖片將要渲染到屏幕時,系統會在主線程對圖片進行解壓縮,而如果圖片已經解壓縮了,系統就不會再對圖片進行解壓縮。因此,也就有了業內的解決方案,在子線程提前對圖片進行強制解壓縮。

而強制解壓縮的原理就是對圖片進行重新繪制,得到一張新的解壓縮后的位圖。其中,用到的最核心的函數是 CGBitmapContextCreate :  

CG_EXTERN CGContextRef __nullable CGBitmapContextCreate(void * __nullable data, size_t width, size_t height, size_t bitsPerComponent, size_t bytesPerRow, CGColorSpaceRef cg_nullable space, uint32_t bitmapInfo) CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);
  • data :如果不為 NULL ,那么它應該指向一塊大小至少為 bytesPerRow * height 字節的內存;如果 為 NULL ,那么系統就會為我們自動分配和釋放所需的內存,所以一般指定 NULL 即可;
  • width 和height :位圖的寬度和高度,分別賦值為圖片的像素寬度和像素高度即可;
  • bitsPerComponent :像素的每個顏色分量使用的 bit 數,在 RGB 顏色空間下指定 8 即可;
  • bytesPerRow :位圖的每一行使用的字節數,大小至少為 width * bytes per pixel 字節。當我們指定 0/NULL 時,系統不僅會為我們自動計算,而且還會進行 cache line alignment 的優化
  • space :就是我們前面提到的顏色空間,一般使用 RGB 即可;
  • bitmapInfo :位圖的布局信息.kCGImageAlphaPremultipliedFirst

五.YYImage/SDWebImage開源框架實現

用于解壓縮圖片的函數 YYCGImageCreateDecodedCopy 存在于 YYImageCoder 類中,核心代碼如下

CGImageRef YYCGImageCreateDecodedCopy(CGImageRef imageRef, BOOL decodeForDisplay) { ... if (decodeForDisplay) { // decode with redraw (may lose some precision)  CGImageAlphaInfo alphaInfo = CGImageGetAlphaInfo(imageRef) & kCGBitmapAlphaInfoMask;  BOOL hasAlpha = NO;  if (alphaInfo == kCGImageAlphaPremultipliedLast ||   alphaInfo == kCGImageAlphaPremultipliedFirst ||   alphaInfo == kCGImageAlphaLast ||   alphaInfo == kCGImageAlphaFirst) {   hasAlpha = YES;  }  // BGRA8888 (premultiplied) or BGRX8888  // same as UIGraphicsBeginImageContext() and -[UIView drawRect:]  CGBitmapInfo bitmapInfo = kCGBitmapByteOrder32Host;  bitmapInfo |= hasAlpha ? kCGImageAlphaPremultipliedFirst : kCGImageAlphaNoneSkipFirst;  CGContextRef context = CGBitmapContextCreate(NULL, width, height, 8, 0, YYCGColorSpaceGetDeviceRGB(), bitmapInfo);  if (!context) return NULL;  CGContextDrawImage(context, CGRectMake(0, 0, width, height), imageRef); // decode  CGImageRef newImage = CGBitmapContextCreateImage(context);  CFRelease(context);  return newImage; } else {  ... }}

它接受一個原始的位圖參數 imageRef ,最終返回一個新的解壓縮后的位圖 newImage ,中間主要經過了以下三個步驟:

  • 使用 CGBitmapContextCreate 函數創建一個位圖上下文;
  • 使用 CGContextDrawImage 函數將原始位圖繪制到上下文中;
  • 使用 CGBitmapContextCreateImage 函數創建一張新的解壓縮后的位圖。

事實上,SDWebImage 中對圖片的解壓縮過程與上述完全一致,只是傳遞給 CGBitmapContextCreate 函數的部分參數存在細微的差別

性能對比:

  • 在解壓PNG圖片,SDWebImage>YYImage
  • 在解壓JPEG圖片,SDWebImage<YYImage

總結

1、圖片文件只有在確認要顯示時,CPU才會對齊進行解壓縮.因為解壓是非常消耗性能的事情.解壓過的圖片就不會重復解壓,會緩存起來.

2、圖片渲染到屏幕的過程: 讀取文件->計算Frame->圖片解碼->解碼后紋理圖片位圖數據通過數據總線交給GPU->GPU獲取圖片Frame->頂點變換計算->光柵化->根據紋理坐標獲取每個像素點的顏色值(如果出現透明值需要將每個像素點的顏色*透明度值)->渲染到幀緩存區->渲染到屏幕

3、面試中如果能按照這個邏輯闡述,應該沒有大的問題.不過,如果細問到離屏渲染和渲染中的細節處理.就需要掌握OpenGL ES/Metal 這個2個圖形處理API. 面試過程可能會遇到不在自己技術能力范圍問題,盡量知之為知之不知為不知.

  • https://github.com/SDWebImage/SDWebImage (本地下載)
  • https://github.com/ibireme/YYImage (本地下載)

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對武林網的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久精品色欧美aⅴ一区二区| 国产精品久久久久高潮| 亚洲精品乱码久久久久久金桔影视| 亚洲美女在线看| 亚洲成人1234| 久久久久久久久久久成人| 成人97在线观看视频| 国产在线观看精品| 亚洲中国色老太| 欧美成人午夜免费视在线看片| 另类视频在线观看| 欧美激情在线播放| 日韩成人在线视频网站| 国产精品户外野外| 69视频在线免费观看| 日韩欧美中文第一页| 午夜精品一区二区三区视频免费看| 日韩精品免费观看| www.久久撸.com| 一区二区三区在线播放欧美| 国产精品xxx视频| 91精品国产成人www| 日韩欧美在线视频日韩欧美在线视频| 成人性生交大片免费观看嘿嘿视频| 国产一区二区三区视频在线观看| 国产日韩av在线| 欧美日韩精品二区| 最近中文字幕mv在线一区二区三区四区| 国产精品劲爆视频| 日韩精品极品在线观看播放免费视频| 欧美一级bbbbb性bbbb喷潮片| 91国产精品视频在线| 亚洲天堂成人在线视频| 亚洲高清一二三区| 亚洲美女激情视频| 亚洲福利视频专区| 日韩大片免费观看视频播放| 国产日韩在线看| 福利视频第一区| 久久久精品日本| 亚洲精品网站在线播放gif| 一夜七次郎国产精品亚洲| 国产精品视频成人| 青青久久aⅴ北条麻妃| 欧美精品一区二区三区国产精品| 国产精品免费一区二区三区都可以| 国内成人精品一区| 98精品在线视频| 欧美成年人网站| 精品欧美激情精品一区| 久久精品成人欧美大片古装| 97在线视频免费| 欧美日韩精品在线观看| 992tv在线成人免费观看| 国产精品国产自产拍高清av水多| 疯狂蹂躏欧美一区二区精品| 久久69精品久久久久久国产越南| 日韩的一区二区| 久久99精品久久久久久青青91| 欧美中文字幕视频| 亚洲国产成人在线视频| 在线观看日韩www视频免费| 欧美日韩ab片| 日韩电影视频免费| 日韩一区二区三区国产| 欧美日韩免费区域视频在线观看| 欧美激情区在线播放| 伊人伊人伊人久久| 国产精品xxxxx| 中文字幕日韩av综合精品| 国产婷婷色综合av蜜臀av| 欧美另类老肥妇| 91精品在线国产| 欧美美女操人视频| 日韩一区二区三区xxxx| 92裸体在线视频网站| 日本一区二区在线播放| 中文字幕精品视频| 三级精品视频久久久久| 久久精品99久久久香蕉| 中文字幕视频在线免费欧美日韩综合在线看| 欧美另类极品videosbestfree| 亚洲女人天堂色在线7777| 国内精品一区二区三区四区| 成人黄色免费在线观看| 91精品久久久久久久久久入口| 日韩中文字幕网| 日日摸夜夜添一区| 亚洲春色另类小说| 4444欧美成人kkkk| 日韩人在线观看| 日韩av免费看| 国产精品永久免费观看| 日韩网站免费观看高清| www高清在线视频日韩欧美| 色综合久久88色综合天天看泰| 欧美成人精品影院| 亚洲国产精品久久久久秋霞蜜臀| 久久久电影免费观看完整版| 欧美视频专区一二在线观看| 日韩在线视频免费观看高清中文| 欧美高清视频免费观看| 成人国产精品久久久久久亚洲| 国内偷自视频区视频综合| 日韩精品在线观看一区| 精品成人av一区| www.xxxx欧美| 国产成人一区三区| 日韩在线视频观看正片免费网站| 亚洲激情视频在线| 这里精品视频免费| 亚洲va久久久噜噜噜久久天堂| 国产有码一区二区| 亚洲激情成人网| 一区二区三区视频在线| 亚洲人成电影网| 精品久久香蕉国产线看观看gif| 日韩性生活视频| 日韩欧美亚洲一二三区| 91麻豆国产精品| 亚洲精品中文字| 91在线视频免费| 欧美性在线观看| 日韩中文字幕在线看| 精品美女久久久久久免费| 国产成人精品在线播放| 亚洲精品一区久久久久久| 亚洲一区美女视频在线观看免费| 欧美一级在线亚洲天堂| 欧美精品免费在线观看| 91成人精品网站| 久久久亚洲欧洲日产国码aⅴ| 日韩av电影国产| 国产女同一区二区| 韩国视频理论视频久久| 另类专区欧美制服同性| 一本色道久久综合亚洲精品小说| 欧美黑人xxxⅹ高潮交| 姬川优奈aav一区二区| 国产欧美日韩中文字幕| 国产精品偷伦免费视频观看的| 欧美性视频在线| 国产成人啪精品视频免费网| xvideos成人免费中文版| 成人做爽爽免费视频| 91精品中文在线| 亚洲一区亚洲二区亚洲三区| 欧美日韩国产中字| 日韩黄色在线免费观看| 国产精品揄拍一区二区| 久久精品精品电影网| 成人综合国产精品| 色婷婷av一区二区三区久久| 亚洲天堂2020| 911国产网站尤物在线观看| 97人人模人人爽人人喊中文字| 欧美激情一区二区三区成人| 中文字幕亚洲精品| 国产日韩中文字幕| 亚洲自拍小视频| 亚洲第一精品久久忘忧草社区| 欧美电影在线免费观看网站| 91精品国产综合久久香蕉922| 国产精品第一视频|