IOS 列表性能優(yōu)化-圖片解碼性能優(yōu)化

前言

為什么需要對圖片進行解碼操作?

事實上,不管是 JPEG 還是 PNG 圖片也祠,都是一種壓縮的位圖圖形格式钾挟。只不過 PNG 圖片是無損壓縮男杈,并且支持 alpha 通道,而 JPEG 圖片則是有損壓縮,可以指定 0-100% 的壓縮比,因此,在將磁盤中的圖片渲染到屏幕之前,必須先要得到圖片的原始像素數(shù)據(jù)跟磨,才能執(zhí)行后續(xù)的繪制操作,這就是為什么需要對圖片解壓縮的原因攒盈。詳見 談談 iOS 中圖片的解壓縮 IOS 中圖片格式問題與性能優(yōu)化 iOS開發(fā):圖片格式與性能優(yōu)化

1.圖片解碼到底有多卡?

測試方法比較簡單抵拘,在一個可以tableView里面展示圖片,圖片是已經(jīng)放在本地的10張圖片,每張圖片大于1MB

代碼如下:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    BannerTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"BannerTableViewCell" forIndexPath:indexPath];
    
    // 獲取圖片
    NSInteger index = 0;
    index = indexPath.row%10;
    NSString *imageName = [NSString stringWithFormat:@"backImage%ld",(long)index];
    //UIImage *image = [UIImage imageNamed:imageName];
    
    NSString *path = [[NSBundle mainBundle] pathForResource:imageName ofType:@"jpg"];
    UIImage *image = [UIImage imageWithContentsOfFile:path];
    
    cell.contentImageView.image = image; 
    return cell;
}

細心的同學可能已經(jīng)注意到了我在代碼中寫了兩種方式加載圖片。
一種是: [UIImage imageNamed:imageName]
一種是: [UIImage imageWithContentsOfFile:path]
后面我再解釋為什么需要對比這兩種加載方式,先上加載的結果吧型豁。

1>使用[UIImage imageWithContentsOfFile:path]

image.png

2>使用[UIImage imageNamed:imageName]

image.png

兩種方式都實際滑動一分鐘, 可以清晰的看到,兩種加載方式一開始都幀數(shù)很低,但是使用imageNamed: 的很快幀數(shù)就恢復到60幀,但是使用imageWithContentsOfFile:會一直卡頓,那是因為使用imageNamed: 會緩存圖片,但是imageWithContentsOfFile: 則不會,而且 使用imageWithContentsOfFile: 出現(xiàn)了明顯的卡頓,出現(xiàn)了明顯的丟幀從曲線上來看能明顯看到兩種方式的差異問題僵蛛。

再來解釋我們使用的兩種加載方式,使用 imageWithContentsOfFile: 實際上是模擬網(wǎng)絡下載圖片到本地后,再從本地加載展示圖片的過程,imageNamed:方式則是模擬從Assets.xcassets 里加載圖片的情況,可以明顯看到蘋果是對從Assets.xcassets 里加載圖片做過優(yōu)化的尚蝌。

2.如何對圖片解碼部分進行優(yōu)化

方案很簡單: 解碼的過程是可以直接放在子線程中的,解碼完成后可以在主線程中將圖片賦值給imageView.image并且緩存下來,下次再次查找到相同的圖片直接在緩存中讀取就可以了墩瞳。
這個過程是不是聽起來很熟悉,是的,這個過程已經(jīng)有很有多的第三方庫實現(xiàn)過了,其中最有名的就是SDWebImage了,SDWebImage的解碼方法是decodedImageWithImage,使用了CGContextDrawImage,有興趣的小伙伴們可以抽空去看看驼壶,在這我就不贅述了,直接上優(yōu)化代碼:

    [self queryImageCache:imageName block:^(UIImage *image) {
        cell.contentImageView.image = image;
    }];

- (void)queryImageCache:(NSString *)filename block:(void(^)(UIImage *image))block
{
    //從內(nèi)存去取喉酌,如果沒取到热凹,就直接讀取文件,在緩存起來
    UIImage *image = [self.memCache objectForKey:filename];
    if(image)
    {
        dispatch_async(dispatch_get_main_queue(), ^{
            if(block)
                block(image);
        });
    }
    else
    {
        //把解壓操作放到子線程
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            NSString *path = [[NSBundle mainBundle] pathForResource:filename ofType:@"jpg"];
            UIImage *image = [UIImage imageWithContentsOfFile:path];
            
            image = [UIImage decodedImageWithImage:image];
            [self.memCache setObject:image forKey:filename];
            
            // 同步主線程
            dispatch_async(dispatch_get_main_queue(), ^{
                if(block)
                    block(image);
            });
       });
    }
}

實驗以上方法后再次進行之前的方法查看FPS和CPU使用情況


image.png
名稱 FPS (平均) CPU(平均) 實驗時間
imageWithContentsOfFile: 47.8 28% 1min
imageNamed: 58.8 10% 1min3
優(yōu)化后 59.9 7% 1min9

可以明顯看到不論是幀數(shù)還是CPU使用情況,優(yōu)化后的列表情況都明顯優(yōu)異多了,雖然這個過程SDWebImage已經(jīng)實現(xiàn)了,但是放在我還是想放在這里來講解下,希望對各位有所幫助泪电。

參考過以下大大的技術博客:
https://blog.ibireme.com/2015/11/12/smooth_user_interfaces_for_ios/
http://www.reibang.com/p/f9ef5dba9ba3?_dt_push=1
http://www.cocoachina.com/cms/wap.php?action=article&id=24599

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末般妙,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子相速,更是在濱河造成了極大的恐慌碟渺,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件突诬,死亡現(xiàn)場離奇詭異苫拍,居然都是意外死亡,警方通過查閱死者的電腦和手機旺隙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進店門绒极,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蔬捷,你說我怎么就攤上這事垄提。” “怎么了周拐?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵铡俐,是天一觀的道長。 經(jīng)常有香客問我妥粟,道長审丘,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任勾给,我火速辦了婚禮滩报,結果婚禮上,老公的妹妹穿的比我還像新娘锦秒。我一直安慰自己露泊,他們只是感情好喉镰,可當我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布旅择。 她就那樣靜靜地躺著,像睡著了一般侣姆。 火紅的嫁衣襯著肌膚如雪生真。 梳的紋絲不亂的頭發(fā)上沉噩,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天,我揣著相機與錄音柱蟀,去河邊找鬼川蒙。 笑死,一個胖子當著我的面吹牛长已,可吹牛的內(nèi)容都是我干的畜眨。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼术瓮,長吁一口氣:“原來是場噩夢啊……” “哼康聂!你這毒婦竟也來了?” 一聲冷哼從身側響起胞四,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤恬汁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后辜伟,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體氓侧,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年导狡,在試婚紗的時候發(fā)現(xiàn)自己被綠了约巷。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡烘豌,死狀恐怖载庭,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情廊佩,我是刑警寧澤囚聚,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站标锄,受9級特大地震影響顽铸,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜料皇,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一谓松、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧践剂,春花似錦鬼譬、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春巩螃,著一層夾襖步出監(jiān)牢的瞬間演怎,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工避乏, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留爷耀,地道東北人。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓拍皮,卻偏偏與公主長得像歹叮,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子铆帽,可洞房花燭夜當晚...
    茶點故事閱讀 42,786評論 2 345

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