Android使用giflib庫(kù)高效加載gif圖片總結(jié)

最近公司項(xiàng)目需要加載大量gif圖片,我們項(xiàng)目用的圖片加載庫(kù)是glide,眾所周知glide自帶加載gif功能,但是真實(shí)使用到項(xiàng)目中 glide加載gif會(huì)占用大量?jī)?nèi)存導(dǎo)致應(yīng)用卡頓,嚴(yán)重的會(huì)奔潰。查看glide源碼發(fā)現(xiàn)glide加載gif圖片,使用java解碼,所以導(dǎo)致內(nèi)存增高。想到原來(lái)項(xiàng)目中用過(guò)的一個(gè)加載本地gif的三方庫(kù)( android-gif-drawable ),人家優(yōu)化的就很好沒(méi)有那么卡,就開(kāi)始研究人家的代碼,研究后發(fā)現(xiàn)人家這個(gè)庫(kù)性能好的原因是他用giflib來(lái)解碼gif,但是他這個(gè)庫(kù)只能加載本地圖片,而我們項(xiàng)目需要加載網(wǎng)絡(luò)圖片,所以就想把glide和giflib做一個(gè)結(jié)合,使用glide下載圖片,bitmap緩存的功能,解碼器替換成giflib,經(jīng)過(guò)一天研究終于成功了,寫文章記錄一下,也希望其他新手朋友有同樣需求的有現(xiàn)成的參考O(∩_∩)O。

使用giflib需要一點(diǎn)ndk開(kāi)發(fā)的經(jīng)驗(yàn),最起碼能看懂cmake語(yǔ)法,因?yàn)楝F(xiàn)在android開(kāi)發(fā)ndk都是使用cmalelist。

ndk部分

首先需要下載 framesequencegiflib(以上網(wǎng)站需要翻墻,請(qǐng)自備梯子)

giflib目錄如下

image

framesequence 項(xiàng)目jin目錄如下

image

按以下步驟操作

1. 使用AndroidStudio創(chuàng)建NDK項(xiàng)目。

2. 把framesequence下的jin文件復(fù)制到自己項(xiàng)目的cpp文件夾下(只復(fù)制我圖片中的.cpp .h文件 多余部分請(qǐng)不要復(fù)制),把giflib文件夾復(fù)制到cpp目錄下。

3. 在cpp文件夾下新增util文件夾,創(chuàng)建log,math的頭文件。(文末會(huì)給demo鏈接,里面有這個(gè)兩個(gè)文件)

image

4. 重新寫CMackList.txt如下

cmake_minimum_required(VERSION 3.4.1)

file(GLOB_RECURSE GIF_LIB ${CMAKE_SOURCE_DIR}/giflib/*.*)
file(GLOB_RECURSE FRAME_SEQUENCE ${CMAKE_SOURCE_DIR}/*.cpp*)

add_library(mygif
        SHARED
        ${FRAME_SEQUENCE}
        ${GIF_LIB})

list(APPEND LIBS
        jnigraphics
        android
        GLESv2
        log
        )

set(LIBS)
list(APPEND LIBS
        jnigraphics
        android
        GLESv2
        log
        )

target_link_libraries(mygif ${LIBS})

最終項(xiàng)目的cpp文件是這樣的

image

這時(shí)候點(diǎn)擊Build -> Refresh LInked C++ Projects

image

等待項(xiàng)目編譯好后,運(yùn)行項(xiàng)目看是否能跑起來(lái),如果跑起來(lái)證明so已生成,關(guān)于ndk的部分就結(jié)束了,剩下只有java代碼了。(●ˇ?ˇ●)

java部分

1. 把framesequence項(xiàng)目中的android文件夾復(fù)制到你自己項(xiàng)目的java文件夾下,如圖

image

2. 項(xiàng)目導(dǎo)入glide(4.+版本),創(chuàng)建一個(gè)類 繼承 AppGlideModule,使用過(guò)glide都知道這個(gè)類的用處,可以生成GlideApp,設(shè)置緩存大小,緩存路徑等功能。我們繼承這個(gè)類主要的目的是替換glide的gif加載


@GlideModule
public class GifGlideModule extends AppGlideModule {

    @Override
    public void registerComponents(@NonNull Context context,
                                   @NonNull Glide glide, @NonNull Registry registry) {
        super.registerComponents(context, glide, registry);
        registry.append(Registry.BUCKET_GIF, InputStream.class,
                FrameSequenceDrawable.class, new GifDecoder(glide.getBitmapPool()));
    }
}

這時(shí)發(fā)現(xiàn)GifDecoder報(bào)錯(cuò),這個(gè)文件是需要我們自己編寫的,代碼如下


public class GifDecoder implements ResourceDecoder<InputStream, FrameSequenceDrawable> {

    private BitmapPool bitmapPool;

    public GifDecoder(BitmapPool bitmapPool) {
        this.bitmapPool = bitmapPool;
    }

    @Override
    public boolean handles(@NonNull InputStream source, @NonNull Options options) throws IOException {
        return true;
    }

    @Nullable
    @Override
    public Resource<FrameSequenceDrawable> decode(@NonNull InputStream source, int width, final int height, @NonNull Options options) throws IOException {
        FrameSequence frameSequence = FrameSequence.decodeStream(source);
        FrameSequenceDrawable frameSequenceDrawable = new FrameSequenceDrawable(frameSequence, new FrameSequenceDrawable.BitmapProvider() {
            @Override
            public Bitmap acquireBitmap(int minWidth, int minHeight) {
                return bitmapPool.get(minWidth, minHeight, Bitmap.Config.ARGB_8888);
            }

            @Override
            public void releaseBitmap(Bitmap bitmap) {
                bitmapPool.put(bitmap);
            }
        });
        return new GifResource(frameSequenceDrawable);
    }
}

GifResource文件也是自己編寫的,代碼如下


public class GifResource extends DrawableResource<FrameSequenceDrawable> {

    public GifResource(FrameSequenceDrawable drawable) {
        super(drawable);
    }

    @NonNull
    @Override
    public Class<FrameSequenceDrawable> getResourceClass() {
        return FrameSequenceDrawable.class;
    }

    @Override
    public int getSize() {
        return 0;
    }

    @Override
    public void recycle() {
        drawable.stop();
        drawable.destroy();
    }
}

到這里 所有的代碼都寫完,重新編譯項(xiàng)目,讓glide生成GlideApp

在代碼中使用

String gif = "gif格式的圖片url";
GlideApp.with(this).as(FrameSequenceDrawable.class).load(gif).into(imageView);

Demo地址

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,797評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,179評(píng)論 3 414
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,628評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,642評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,444評(píng)論 6 405
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,948評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,040評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,185評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,717評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,602評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,794評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,316評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,045評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,418評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,671評(píng)論 1 281
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,414評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,750評(píng)論 2 370

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