美文网首页
Universal-Image-Loader源码解析(1)总体框

Universal-Image-Loader源码解析(1)总体框

作者: Rocky1982 | 来源:发表于2017-04-12 17:11 被阅读80次

    Universal-Image-Loader(以下简称UIL)这个Android的图片加载库年代已经比较久远了,现在作者也不更新了,不过使用的人还是很多,源码还是值得一看的。我打算分几章来分析一下UIL的源码。第一章就先介绍一下总体框架。

    整体流程

    首先借用官方的一张图看看UIL加载图片的流程,这对后续的分析起到一定的指导作用,大家可以将各个模块按照这幅流程图对号入座,这样就对模块的整体功能一目了然了。

    Class Diagram-Overview.jpg

    整体类图

    UIL的使用方法这里就不介绍了,网上的教程很多。这里直接开始看看整体类图。

    Class Diagram-Overview.jpg

    类图里只画出了最重要的一些类和接口。下面我就针对这些类做一些大概的说明,详细的介绍在后续章节里奉上。

    ImageLoader

    ImageLoade在UIL中相当于指挥官的角色。大部分加载图片的功能都只需要用到这一个类就够了。比如:

    ImageLoader imageLoader = ImageLoader.getInstance(); // Get singleton instance
    imageLoader.displayImage(imageUri, imageView);
    

    就是这样简单,只需要ImageLoader 一个类以及图片的uri和ImageView两个参数就够了。
    ImageLoader 用的是单例模式,它里面重写了多个displayImage和loadImage方法。displayImage方法用来将图片展示在ImageView上,loadImage只是加载图片而不展示。所有加载图片的操作都是通过ImageLoader 的这两类方法完成的。
    ImageLoader里包含一个ImageLoaderConfiguration对象,这个对象通过init方法传入。ImageLoaderConfiguration就是一些UIL的配置,它使用了Builder模式来创建,比较简单,略过。

    public synchronized void init(ImageLoaderConfiguration configuration) {
            if (configuration == null) {
                throw new IllegalArgumentException(ERROR_INIT_CONFIG_WITH_NULL);
            }
            if (this.configuration == null) {
                L.d(LOG_INIT_CONFIG);
                engine = new ImageLoaderEngine(configuration);
                this.configuration = configuration;
            } else {
                L.w(WARNING_RE_INIT_CONFIG);
            }
        }
    

    在init方法中还创建了ImageLoaderEngine这个类。这个类比较重要,正如它名字所描述的那样,它就是UIL的任务引擎。ImageLoader加载图片的逻辑是先判断MemoryCache中有没有缓存的图片,如果有且不需要后处理,就直接显示,需要处理就创建ProcessAndDisplayImageTask,然后提交到Engine。如果没有就创建LoadAndDisplayImageTask,然后提交到Engine中执行的。下面是displayImage方法的一个片段。

    Bitmap bmp = configuration.memoryCache.get(memoryCacheKey);
            if (bmp != null && !bmp.isRecycled()) {
                ...
    
                if (options.shouldPostProcess()) {
                    ...
                    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);
                    ...
                }
            } else {
                ...
                LoadAndDisplayImageTask displayTask = new LoadAndDisplayImageTask(engine, imageLoadingInfo,
                        defineHandler(options));
                if (options.isSyncLoading()) {
                    displayTask.run();
                } else {
                    engine.submit(displayTask);
                }
            }
    

    ImageLoaderEngine

    从类图中可以看到,有两类Task会提交到Engine中执行,他们分别是ProcessAndDisplayImageTask和LoadAndDisplayImageTask。ProcessAndDisplayImageTask的任务是加载图片并做后处理后显示图片,LoadAndDisplayImageTask只是单纯的加载和显示图片。
    在类图中还有一个类:DisplayBitmapTask。它的任务就是显示图片。但在Engine中并没有一个submit方法是用来提交DisplayBitmapTask的。这个类其实是在前两个Task类的Run方法中用来加载图片的。补充一句,这三种Task都实现了Runnable接口。
    Engine执行Task是提交到Executor中执行的。(Executor的知识请查阅Java文档)看一下代码片段:

    private Executor taskExecutor;
    private Executor taskExecutorForCachedImages;
    private Executor taskDistributor;
    

    一共有三个Executor。

    void submit(final LoadAndDisplayImageTask task) {
            taskDistributor.execute(new Runnable() {
                @Override
                public void run() {
                    File image = configuration.diskCache.get(task.getLoadingUri());
                    boolean isImageCachedOnDisk = image != null && image.exists();
                    initExecutorsIfNeed();
                    if (isImageCachedOnDisk) {
                        taskExecutorForCachedImages.execute(task);
                    } else {
                        taskExecutor.execute(task);
                    }
                }
            });
        }
    

    taskDistributor对Task进行分发。已经缓存在Disk上的图片就交给taskExecutorForCachedImages执行,没有缓存的就交给taskExecutor。

    BitmapDisplayer和ImageAware

    这是两个接口。他们都和显示图片相关。

     
    public interface BitmapDisplayer {
        /**
         *
         * @param bitmap     Source bitmap
         * @param imageAware {@linkplain com.nostra13.universalimageloader.core.imageaware.ImageAware Image aware view} to
         *                   display Bitmap
         * @param loadedFrom Source of loaded image
         */
        void display(Bitmap bitmap, ImageAware imageAware, LoadedFrom loadedFrom);
    }
    
    public interface ImageAware {
        int getWidth();
        int getHeight();
        ViewScaleType getScaleType();
        View getWrappedView();
        boolean isCollected();
        int getId();
    
        boolean setImageDrawable(Drawable drawable);
    
        boolean setImageBitmap(Bitmap bitmap);
    }
    

    将图片的显示分成两个接口体现出了mvc的思想。BitmapDisplayer负责处理图片数据:Bitmap,ImageAware负责显示。BitmapDisplayer的实现类有:CircleBitmapDisplayer, FadeInBitmapDisplayer, RoundedBitmapDisplayer, RoundedVignetteBitmapDisplayer, SimpleBitmapDisplayer。从这些名字就基本可以看出这些类的功能。比如CircleBitmapDisplayer:

    public class CircleBitmapDisplayer implements BitmapDisplayer {
        ...
        @Override
        public void display(Bitmap bitmap, ImageAware imageAware, LoadedFrom loadedFrom) {
            if (!(imageAware instanceof ImageViewAware)) {
                throw new IllegalArgumentException("ImageAware should wrap ImageView. ImageViewAware is expected.");
            }
    
            imageAware.setImageDrawable(new CircleDrawable(bitmap, strokeColor, strokeWidth));
        }
    
        public static class CircleDrawable extends Drawable {
            ...
            @Override
            public void draw(Canvas canvas) {
                canvas.drawCircle(radius, radius, radius, paint);
                if (strokePaint != null) {
                    canvas.drawCircle(radius, radius, strokeRadius, strokePaint);
                }
            }
            ...
        }
    }
    

    它的功能就是在将原始图片变为圆形且有个圆圈的Drawable。
    ImageAware的直接实现只有一个类:ViewAware。我们从ImageAware的接口总可以看出,它只是提供一些显示图片的基本数据,应该是View的一个封装。ViewAware就印证了这种想法。其中一点比较重要的是ViewAware使用了弱引用来封装View。

    public abstract class ViewAware implements ImageAware {
        ...
        protected Reference<View> viewRef;
        ...
    }
    

    ViewAware的子类有两个ImageViewAware和NonViewAware。ImageViewAware就是ImageView的封装。它用ImageView来显示图片。当我们调用各种displayImage方法时,最终都会用ImageViewAware来显示图片。

    @Override
        protected void setImageDrawableInto(Drawable drawable, View view) {
            ((ImageView) view).setImageDrawable(drawable);
            if (drawable instanceof AnimationDrawable) {
                ((AnimationDrawable)drawable).start();
            }
        }
    
        @Override
        protected void setImageBitmapInto(Bitmap bitmap, View view) {
            ((ImageView) view).setImageBitmap(bitmap);
        }
    

    ImageViewAware简单的将显示图片的任务交给了ImageView。

    NonViewAware里面封装的View为null,当调用loadImage方法时,最终都会用到NonViewAware。

        @Override
        public boolean setImageDrawable(Drawable drawable) { // Do nothing
            return true;
        }
    
        @Override
        public boolean setImageBitmap(Bitmap bitmap) { // Do nothing
            return true;
        }
    

    因为loadImage只是为了获取图片数据,而不用显示,所以NonViewAware的setImageDrawable和setImageBitmap只是简单的返回true。

    MemoryCache和DiskCache

    图片的缓存就在这里了。后面的章节会详细介绍,这里略过。

    总结

    UIL的整体结构还是很清晰的。下面是包图,基本上主要的内容在上面都介绍过了。对某一方面有兴趣的朋友可以去相关包下面查看源码。


    packages.png

    相关文章

      网友评论

          本文标题:Universal-Image-Loader源码解析(1)总体框

          本文链接:https://www.haomeiwen.com/subject/lwjmattx.html