Android 图片缓存框架-Universal Image Loader 源码分析(上篇)

来源网络.png

不知道大家是不是注册了github,结果大部分的时间是在star项目,最多需要的时候百度个使用指南,代码复制粘贴,然后直接用一下开源项目。

有个智者说过:不看源码的程序员不是个好程序员

如果说有Android的开源框架是非分析不可的,那肯定就是Universal Image Loader后面简称UIL)。因为无数人推荐过需要看看它的源码,所以我也就开始我的源码计划。

按照惯例,需要把github地址给上>>>>Universal Image Loader <<<<狂戳左边的这个链接。

1. 项目简介

1.1 UIL简介

UIL是一个什么开源项目?按照Github上面的解释

UIL是一个功能强大、灵活的、可以高度自定义化的组件,它可以实现图片加载,图片缓存和图片显示。

简单来说,它就是可以实现我们异步和同步加载图片,同时能够做到图片的三级缓存。至于哪三级(不是你想的三级啊!),后面一一解密。

1.2 项目结构

项目分包.png

首先我们从这个项目的分包开始看起。参照上图,我们看到UIL主要分了三个包

  • cache 缓存模块:包含内存缓存、磁盘缓存
  • core 核心功能模块:包含图片解码、图片下载、图片显示、图片处理、图片操作任务以及一些配置模块等
  • utils 工具模块:日志、图片大小等一些工具类
Structure.png

主要的几大模块就是上面这个图所展示的,我们从我们最熟悉的ImageLoader类开始分析一下。

1.3 UIL 简单使用步骤

ImageLoader使用方法,最简单的就是以下几个步骤

  1. getInstance 获得ImageLoader的单例对象
  2. init 初始化它的配置信息
  3. displayImage / loadImage 输入图片路径和对应控件
// 1. 先获取一个单例
ImageLoader imageLoader = ImageLoader.getInstance(); 

// 2. 给它配置它的多个参数(以下代码来自于UIL 自带的Sample)
ImageLoaderConfiguration.Builder config = new ImageLoaderConfiguration.Builder(context);
config.threadPriority(Thread.NORM_PRIORITY - 2);
config.denyCacheImageMultipleSizesInMemory();
config.diskCacheFileNameGenerator(new Md5FileNameGenerator());
config.diskCacheSize(50 * 1024 * 1024); // 50 MiBconfig.tasksProcessingOrder(QueueProcessingType.LIFO);
config.writeDebugLogs(); // Remove for release app
imageLoader.init(config.build());

显示的方式有三种,大家看下面的代码

// 3. 显示指定的Imageurl在相应ImageView上
imageLoader.displayImage(imageUri, imageView);

// 3. 也可以使用loadImage 方法,使用回调函数进行图片异步处理
imageLoader.loadImage(imageUri, new SimpleImageLoadingListener(){ 
  @Override 
  public void onLoadingComplete(String imageUri, View view, Bitmap loadedImage) { 
    // Do whatever you want with Bitmap 
  }
});

// 3. 同步加载
Bitmap bmp = imageLoader.loadImageSync(imageUri);

在这三个步骤结束后,你就不需要管图片的加载问题,UIL就会自动跟你加载的对应地址处的图片。

那我们的模块分析就从ImageLoader这几个初始化步骤开始吧。

2 模块分析

2.1 ImageLoader模块

前两个步骤,getInstance就是获取一个单例,没什么好特别讲的。init也就是将Config对象赋值给内部的变量。我们从displayImage方法开始研究。

2.1.1 displayImage方法

loadImage和displayImage最后都会使用displayImage

我们来看displayImage最终的方法的实现(为了我们专注主要功能,我刨去了一些检测null的代码,只留下主要功能!)

    public void displayImage(String uri, ImageAware imageAware, DisplayImageOptions options,
            ImageSize targetSize, ImageLoadingListener listener, ImageLoadingProgressListener progressListener) {
        ......
        // 判断一下各项参数是不是空
        // 如果Uri是空,那么就直接完成加载过程。

        String memoryCacheKey = MemoryCacheUtils.generateKey(uri, targetSize);
        engine.prepareDisplayTaskFor(imageAware, memoryCacheKey);

        listener.onLoadingStarted(uri, imageAware.getWrappedView());

        Bitmap bmp = configuration.memoryCache.get(memoryCacheKey);
        if (bmp != null && !bmp.isRecycled()) {
            if (options.shouldPostProcess()) {
                ImageLoadingInfo imageLoadingInfo = new ImageLoadingInfo(uri, imageAware, targetSize, memoryCacheKey,
                        options, listener, progressListener, engine.getLockForUri(uri));
                ProcessAndDisplayImageTask displayTask = new ProcessAndDisplayImageTask(engine, bmp, imageLoadingInfo,
                        defineHandler(options));
                if (options.isSyncLoading()) {
                    displayTask.run();
                } else {
                    engine.submit(displayTask);
                }
            } else {
                options.getDisplayer().display(bmp, imageAware, LoadedFrom.MEMORY_CACHE);
                listener.onLoadingComplete(uri, imageAware.getWrappedView(), bmp);
            }
        } else {
            ...

            ImageLoadingInfo imageLoadingInfo = new ImageLoadingInfo(uri, imageAware, targetSize, memoryCacheKey,
                    options, listener, progressListener, engine.getLockForUri(uri));
            LoadAndDisplayImageTask displayTask = new LoadAndDisplayImageTask(engine, imageLoadingInfo,
                    defineHandler(options));
            if (options.isSyncLoading()) {
                displayTask.run();
            } else {
                engine.submit(displayTask);
            }
        }
    }

总的来说这个流程为以下几个步骤:

  1. 生成指定的Key,将其与控件ImageAware一一对应
  2. 从内存缓存中获取指定Key的Bitmap
  3. 如果内存中没有Bitmap就通过engine启动LoadAndDisplayImageTask(这是个Runnable)
  4. 如果内存中有Bitmap那么就通过engine启动ProcessAndDisplayImageTask(这也是个Runnable)
displayImage流程图.png

下面我们来看一下这两个Task具体做些什么。

2.1.2 ProcessAndDisplayImageTask

这个Task的注释解释道,这是个处理并且显示图片的任务,显示任务的工作交给DisplayImageTask去做。因为这是一个Runnable对象。所以让我们来看一下它的run方法。

    public void run() {
        BitmapProcessor processor = imageLoadingInfo.options.getPostProcessor();
        Bitmap processedBitmap = processor.process(bitmap);
        DisplayBitmapTask displayBitmapTask = new DisplayBitmapTask(processedBitmap, imageLoadingInfo, engine,
                LoadedFrom.MEMORY_CACHE);
        LoadAndDisplayImageTask.runTask(displayBitmapTask, imageLoadingInfo.options.isSyncLoading(), handler, engine);
    }

简单的来说,这个ProcessAndDisplayImageTask做的工作是以下四步:

  1. 从ImageLoadingInfo的DisplayImageOption拿到BitmapProcess,
    getPostProcessor这个方法是取得一个在显示Display之前,在保存到MemoryCache之后的BitmapProcessor
  2. 从BitmapProcess中调用process方法处理Bitmap
  3. 新建一个DisplayBitmapTask,然后丢进静态方法runTask中去执行。

runTask实现的工作,主要是Handler.post(Runnable),具体的细节看源码

总结来说ProcessAndDisplayImageTask主要是起到一个联合的过程,负责将BitmapProcessor和DisplayBitmapTask的按照顺序连接到一起。

2.1.3 LoadAndDisplayImageTask

前者ProcessAndDisplayImageTask主要是在内存缓存中有的时候执行的Task,那么在内存缓存中没有对应Bitmap的时候,LoadAndDisplayImageTask所执行的事务就会相对复杂一些。

让我们也从run方法开始看起。

    @Override
    public void run() {
        if (waitIfPaused()) return;
        if (delayIfNeed()) return;

        ReentrantLock loadFromUriLock = imageLoadingInfo.loadFromUriLock;
        ...
        loadFromUriLock.lock();
        Bitmap bmp;
        try {
            ...
            bmp = configuration.memoryCache.get(memoryCacheKey);
            if (bmp == null || bmp.isRecycled()) {
                bmp = tryLoadBitmap();
                if (bmp == null) return; // listener callback already was fired
                ...

                if (options.shouldPreProcess()) {
                    bmp = options.getPreProcessor().process(bmp);
                    ...
                }

                if (bmp != null && options.isCacheInMemory()) {
                    configuration.memoryCache.put(memoryCacheKey, bmp);
                }
            } else {
                loadedFrom = LoadedFrom.MEMORY_CACHE;
            }

            if (bmp != null && options.shouldPostProcess()) {
                bmp = options.getPostProcessor().process(bmp);
                ...
            }
            ...
        } catch (TaskCancelledException e) {
            fireCancelEvent();
            return;
        } finally {
            loadFromUriLock.unlock();
        }

        DisplayBitmapTask displayBitmapTask = new DisplayBitmapTask(bmp, imageLoadingInfo, engine, loadedFrom);
        runTask(displayBitmapTask, syncLoading, handler, engine);
    }

这个函数的执行过程有点长,我们一步一步来

  1. 这个线程的执行会判断是否需要等待,或者是否需要延迟对应时间。具体细节我们后面再分析。
if (waitIfPaused()) return; 
if (delayIfNeed()) return;
  1. 通过重入锁ReentrantLock 进行图片加载过程的锁定

  2. 先从内存缓存中获取对应Key值的Bitmap

bmp = configuration.memoryCache.get(memoryCacheKey);
  1. 如果内存中没有读取到,那么就调用tryLoadBitmap继续从其他地方读取Bitmap,如果tryLoadBitmap没有能读取到Bitmap,那么就结束这次Task。如果读取到就继续下一步(tryLoadBitmap的过程我们后面再说)

  2. 对于不是从内存获取到的Bitmap对象,我们需要做两步,首先获取一个PreProcessor,也就是在Bitmap对象存入缓存前的BitmapProcessor(跟之前的PostProcessor类似),将图片进行一个预处理。
    如果读取到了Bitmap,那么我们就将其他方式读取的Bitmap重新放入内存缓存中,这样方便下次进行这个图片获取时,能更加高效。因为图片的读取,首先是从内存中获取的。

if (options.shouldPreProcess()) {
        bmp = options.getPreProcessor().process(bmp);
        ...
}
if (bmp != null && options.isCacheInMemory()) {
       configuration.memoryCache.put(memoryCacheKey, bmp);
}
  1. 最后将bmp 也就是不管是从内存中读取的,还是通过tryLoadBitmap方法读取的,进行一个PostProcess的处理,也就是显示预处理。

  2. 最后执行DisplayBitmapTask,控件对应的Bitmap显示到控件上去。


-tryLoadBitmap

之前说到了UIL的缓存机制是采用了三级缓存机制。之前我们在LoadAndDisplayImageTask中,首先会从memoryCache中加载

这个就是三级缓存机制中的第一级——内存

第二级和第三极就是在tryLoadBitmap中实现的。让我们进代码中看一下

private Bitmap tryLoadBitmap() throws TaskCancelledException {
        Bitmap bitmap = null;
        
        File imageFile = configuration.diskCache.get(uri);
        if (imageFile != null && imageFile.exists() && imageFile.length() > 0) {
            loadedFrom = LoadedFrom.DISC_CACHE;
            ...
            bitmap = decodeImage(Scheme.FILE.wrap(imageFile.getAbsolutePath()));
        }
        if (bitmap == null || bitmap.getWidth() <= 0 || bitmap.getHeight() <= 0) {
            ...
            loadedFrom = LoadedFrom.NETWORK;

            String imageUriForDecoding = uri;
            if (options.isCacheOnDisk() && tryCacheImageOnDisk()) {
                imageFile = configuration.diskCache.get(uri);
                if (imageFile != null) {
                    imageUriForDecoding = Scheme.FILE.wrap(imageFile.getAbsolutePath());
                }
            }
            ...
            bitmap = decodeImage(imageUriForDecoding);

            if (bitmap == null || bitmap.getWidth() <= 0 || bitmap.getHeight() <= 0) {
                fireFailEvent(FailType.DECODING_ERROR, null);
            }
        }
        return bitmap;
    }

我们看到会调用Configuration的diskCache.get(uri)方法,获取磁盘缓存中的获取对应的File,

这个就是三级缓存机制中的第二级——磁盘

继续往下分析,由于磁盘缓存是用File对象进行映射的,这里需要调用decodeImage方法将File对应的图片解码成Bitmap对象。当磁盘中还是没有这个图片,那么显然需要去网络上下载。

通过tryCacheImageOnDisk方法将网络上的资源,缓存到磁盘中。然后再从DiskCache中去取出File对象,继续解码成Bitmap对象。最后成功就返回。


-tryCacheImageOnDisk 缓存图片到Disk中

在磁盘中没有读取到,UIL就需要从网络上去请求资源了。

这就是UIL的三级缓存机制中的第三级——网络请求

我们看代码:

private boolean tryCacheImageOnDisk() throws TaskCancelledException {
    ...
    boolean loaded;
    ...
    loaded = downloadImage();
    if (loaded) {
        int width = configuration.maxImageWidthForDiskCache;
        int height = configuration.maxImageHeightForDiskCache;
        if (width > 0 || height > 0) {
            ...
            resizeAndSaveImage(width, height); // TODO : process boolean result
        }
    }
    ...
    return loaded;
}

调用downloadImage方法下载图片,然后对图片进行长宽的处理,设置为缓存中的最大值再存入磁盘缓存DiskCache中。

而downloadImage主要就是调用ImageDownloader进行数据的下载,然后保存到DiskCache中。这个ImageDownloader具体细节我们后面具体分析。这里我们只要知道它是一个下载器的接口,可以自行实现将Uri网络请求得到图片流。

2.1.4 小结

到这里我们对ImageLoader的一个基本流程大概介绍完了,我们来小结一下。

UIL的整个处理逻辑从ImageLoader这个类的displayImage方法开始, 它主要工作就是在内存中获取对应Key值的Bitmap,然后通过获取结果,丢给ImageLoaderEngine线程池内去执行不同类型的任务,其中主要两个任务就是ProcessAndDisplayImageTaskLoadAndDisplayImageTask

前者是内存有就Bitmap就直接通过BitmapProcessor处理然后丢给DisplayTask任务去显示。后者是没有内存的时候就需要开启三级缓存机制,进行逐级的加载。

三级缓存指的是:内存、磁盘、网络

分别通过MemoryCache、DiskCache和ImageDownloader三个去实现了。


总结

第一次写源码的分析,有什么不足请批评指正。

后面还会继续跟进分析一下UIL的缓存的实现方式,个人觉得这个模块还是蛮有意思的。

同时我觉得源码嘛,还是要自己对着代码看,光看这个文章还是会一头雾水的。而且不要陷于代码的细节,执着于某个细节是怎么实现的,忽略了主干的分析,最后看得会很头大的。

这里安利之前看到的介绍代码阅读的文章,《代码阅读的姿势

纸上得来终觉浅,绝知此事要躬行


参考文章:

  1. Android Universal Image Loader 源码分析
  1. Android Universal Image Loader Github readme

尊重原创,转载请注明出处

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容