拓展-iOS 中圖片的解壓縮到渲染過程

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

image

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

  • CPU: 計(jì)算視圖frame捂人,圖片解碼,需要繪制紋理圖片通過數(shù)據(jù)總線交給GPU
  • GPU: 紋理混合,頂點(diǎn)變換與計(jì)算,像素點(diǎn)的填充計(jì)算,渲染到幀緩沖區(qū)趣倾。
  • 時(shí)鐘信號:垂直同步信號V-Sync / 水平同步信號H-Sync刁品。
  • iOS設(shè)備雙緩沖機(jī)制:顯示系統(tǒng)通常會引入兩個(gè)幀緩沖區(qū)印蔬,雙緩沖機(jī)制

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

對應(yīng)應(yīng)用來說,圖片是最占用手機(jī)內(nèi)存的資源基显,將一張圖片從磁盤中加載出來蘸吓,并最終顯示到屏幕上,中間其實(shí)經(jīng)過了一系列復(fù)雜的處理過程撩幽。

二.圖片加載的工作流程

  1. 假設(shè)我們使用 +imageWithContentsOfFile: 方法從磁盤中加載一張圖片库继,這個(gè)時(shí)候的圖片并沒有解壓縮;

  2. 然后將生成的 UIImage 賦值給 UIImageView 窜醉;

  3. 接著一個(gè)隱式的 CATransaction 捕獲到了 UIImageView 圖層樹的變化宪萄;

  4. 在主線程的下一個(gè) runloop 到來時(shí),Core Animation 提交了這個(gè)隱式的 transaction 榨惰,這個(gè)過程可能會對圖片進(jìn)行 copy 操作拜英,而受圖片是否字節(jié)對齊等因素的影響,這個(gè) copy 操作可能會涉及以下部分或全部步驟:

    • 分配內(nèi)存緩沖區(qū)用于管理文件 IO 和解壓縮操作琅催;
    • 將文件數(shù)據(jù)從磁盤讀到內(nèi)存中居凶;
    • 將壓縮的圖片數(shù)據(jù)解碼成未壓縮的位圖形式,這是一個(gè)非常耗時(shí)的 CPU 操作藤抡;
    • 最后 Core AnimationCALayer使用未壓縮的位圖數(shù)據(jù)渲染 UIImageView 的圖層侠碧。
    • CPU計(jì)算好圖片的Frame,對圖片解壓之后.就會交給GPU來做圖片渲染
  5. 渲染流程

    • GPU獲取獲取圖片的坐標(biāo)
    • 將坐標(biāo)交給頂點(diǎn)著色器(頂點(diǎn)計(jì)算)
    • 將圖片光柵化(獲取圖片對應(yīng)屏幕上的像素點(diǎn))
    • 片元著色器計(jì)算(計(jì)算每個(gè)像素點(diǎn)的最終顯示的顏色值)
    • 從幀緩存區(qū)中渲染到屏幕上

我們提到了圖片的解壓縮是一個(gè)非常耗時(shí)的 CPU 操作,并且它默認(rèn)是在主線程中執(zhí)行的缠黍。那么當(dāng)需要加載的圖片比較多時(shí)弄兜,就會對我們應(yīng)用的響應(yīng)性造成嚴(yán)重的影響,尤其是在快速滑動的列表上,這個(gè)問題會表現(xiàn)得更加突出替饿。

三.為什么要解壓縮圖片

既然圖片的解壓縮需要消耗大量的 CPU 時(shí)間语泽,那么我們?yōu)槭裁催€要對圖片進(jìn)行解壓縮呢?是否可以不經(jīng)過解壓縮盛垦,而直接將圖片顯示到屏幕上呢湿弦?答案是否定的。要想弄明白這個(gè)問題腾夯,我們首先需要知道什么是位圖

其實(shí)颊埃,位圖就是一個(gè)像素?cái)?shù)組,數(shù)組中的每個(gè)像素就代表著圖片中的一個(gè)點(diǎn)蝶俱。我們在應(yīng)用中經(jīng)常用到的 JPEG 和 PNG 圖片就是位圖

大家可以嘗試

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

打印rawData,這里就是圖片的原始數(shù)據(jù).

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

// return image as PNG. May return nil if image has no CGImageRef or invalid bitmap format
UIKIT_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);

因此竿拆,在將磁盤中的圖片渲染到屏幕之前宙拉,必須先要得到圖片的原始像素?cái)?shù)據(jù),才能執(zhí)行后續(xù)的繪制操作丙笋,這就是為什么需要對圖片解壓縮的原因谢澈。

四.解壓縮原理

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

我們前面已經(jīng)提到了怠肋,當(dāng)未解壓縮的圖片將要渲染到屏幕時(shí)敬鬓,系統(tǒng)會在主線程對圖片進(jìn)行解壓縮,而如果圖片已經(jīng)解壓縮了笙各,系統(tǒng)就不會再對圖片進(jìn)行解壓縮钉答。因此,也就有了業(yè)內(nèi)的解決方案酪惭,在子線程提前對圖片進(jìn)行強(qiáng)制解壓縮希痴。

而強(qiáng)制解壓縮的原理就是對圖片進(jìn)行重新繪制者甲,得到一張新的解壓縮后的位圖春感。其中,用到的最核心的函數(shù)是 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 ,那么它應(yīng)該指向一塊大小至少為 bytesPerRow * height 字節(jié)的內(nèi)存鲫懒;如果 為 NULL 嫩实,那么系統(tǒng)就會為我們自動分配和釋放所需的內(nèi)存,所以一般指定 NULL 即可窥岩;
  • width 和height :位圖的寬度和高度甲献,分別賦值為圖片的像素寬度和像素高度即可;
  • bitsPerComponent :像素的每個(gè)顏色分量使用的 bit 數(shù)颂翼,在 RGB 顏色空間下指定 8 即可晃洒;
  • bytesPerRow :位圖的每一行使用的字節(jié)數(shù),大小至少為 width * bytes per pixel 字節(jié)朦乏。當(dāng)我們指定 0/NULL 時(shí)球及,系統(tǒng)不僅會為我們自動計(jì)算,而且還會進(jìn)行cache line alignment 的優(yōu)化
  • space :就是我們前面提到的顏色空間呻疹,一般使用 RGB 即可吃引;
  • bitmapInfo :位圖的布局信息.kCGImageAlphaPremultipliedFirst

五.YYImage\SDWebImage開源框架實(shí)現(xiàn)

  • 用于解壓縮圖片的函數(shù) 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 {
        ...
    }
}

它接受一個(gè)原始的位圖參數(shù) imageRef 刽锤,最終返回一個(gè)新的解壓縮后的位圖 newImage 镊尺,中間主要經(jīng)過了以下三個(gè)步驟:

  • 使用 CGBitmapContextCreate 函數(shù)創(chuàng)建一個(gè)位圖上下文;
  • 使用 CGContextDrawImage 函數(shù)將原始位圖繪制到上下文中并思;
  • 使用 CGBitmapContextCreateImage 函數(shù)創(chuàng)建一張新的解壓縮后的位圖庐氮。

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

性能對比:
  • 在解壓PNG圖片,SDWebImage>YYImage
  • 在解壓JPEG圖片,SDWebImage<YYImage

總結(jié)

  1. 圖片文件只有在確認(rèn)要顯示時(shí),CPU才會對齊進(jìn)行解壓縮.因?yàn)榻鈮菏欠浅O男阅艿氖虑?解壓過的圖片就不會重復(fù)解壓,會緩存起來.

  2. 圖片渲染到屏幕的過程: 讀取文件->計(jì)算Frame->圖片解碼->解碼后紋理圖片位圖數(shù)據(jù)通過數(shù)據(jù)總線交給GPU->GPU獲取圖片F(xiàn)rame->頂點(diǎn)變換計(jì)算->光柵化->根據(jù)紋理坐標(biāo)獲取每個(gè)像素點(diǎn)的顏色值(如果出現(xiàn)透明值需要將每個(gè)像素點(diǎn)的顏色*透明度值)->渲染到幀緩存區(qū)->渲染到屏幕

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

https://github.com/SDWebImage/SDWebImage
https://github.com/ibireme/YYImage

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末旭愧,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子宙暇,更是在濱河造成了極大的恐慌输枯,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件占贫,死亡現(xiàn)場離奇詭異桃熄,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)型奥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進(jìn)店門瞳收,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人厢汹,你說我怎么就攤上這事螟深。” “怎么了烫葬?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵界弧,是天一觀的道長凡蜻。 經(jīng)常有香客問我,道長垢箕,這世上最難降的妖魔是什么划栓? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮条获,結(jié)果婚禮上忠荞,老公的妹妹穿的比我還像新娘。我一直安慰自己帅掘,他們只是感情好委煤,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著修档,像睡著了一般素标。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上萍悴,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天头遭,我揣著相機(jī)與錄音,去河邊找鬼癣诱。 笑死计维,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的撕予。 我是一名探鬼主播鲫惶,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼实抡!你這毒婦竟也來了欠母?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤吆寨,失蹤者是張志新(化名)和其女友劉穎赏淌,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體啄清,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡六水,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了辣卒。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片掷贾。...
    茶點(diǎn)故事閱讀 40,040評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖荣茫,靈堂內(nèi)的尸體忽然破棺而出想帅,到底是詐尸還是另有隱情,我是刑警寧澤啡莉,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布港准,位于F島的核電站憎乙,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏叉趣。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一该押、第九天 我趴在偏房一處隱蔽的房頂上張望疗杉。 院中可真熱鬧,春花似錦蚕礼、人聲如沸烟具。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽朝聋。三九已至,卻和暖如春囤躁,著一層夾襖步出監(jiān)牢的瞬間冀痕,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工狸演, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留言蛇,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓宵距,卻偏偏與公主長得像腊尚,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子满哪,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評論 2 355

推薦閱讀更多精彩內(nèi)容