跟着我一步步写一个图片加载框架

起因

对于Android开发来说, 图片加载框架相信很多人都用过, 比如Android-Universal-Image-Loader, Picasso, Glide等等, 但是如果能够自己去实现一个图片加载框架, 那对于充分理解图片的三级缓存(三级还是二级?emmm...其实这并不重要)是十分有帮助的,同时也能让我们更好的理解那些知名的图片加载框架的原理。下面我将演示如何写一个简易的图片加载框架。


模块结构

首先说明一下框架的整体结构。
这是我定义的ImageLoader的结构,里面含有两个成员变量:MemoryCache和DiskCache,是我自定义的两个类。MemoryCache负责管理内存中的bitmap加载。DiskCache负责磁盘的bitmap加载,同时DiskCache中有一个ImageFetcher的模块,负责从网络下载图片到本地磁盘。


加载流程

现在来梳理一下图片的加载流程。



具体就是如上面的流程图所示,已经画的比较详细了。首先ImageLoader根据图片的url从MemoryCache中取bitmap,如果MemoryCache中没有,就去DiskCache中取bitmap。DiskCache根据图片url先从磁盘取bitmap,如果没有就从网络下载图片到磁盘,然后再从磁盘取出这张bitmap返回给ImageLoader。最后就是将这张bitmap设置到ImageView上。接下来就是具体的细节代码了。


ImageLoader

ImageLoader中主要的是displayImage(String url, ImageView iv)这个方法

public void displayImage(final String url, final ImageView imageView, final int reqWidth, final int reqHeight) {
    imageView.setTag(url);
        Bitmap bitmap = mMemoryCache.get(url);
        if (bitmap != null) {
            imageView.setImageBitmap(bitmap);
            return;
        }
        Runnable loadBitmapRunnable = new Runnable() {
            @Override
            public void run() {
                // 从本地或者网络获取图片, 在子线程中进行
                final Bitmap bitmap = mDiskCache.get(url, reqWidth, reqHeight);
                if (bitmap == null) {
                    return;
                }
                // 添加到内存缓存中
                mMemoryCache.put(url, bitmap);
                mHandler.post(new Runnable() {
                    @Override
                    public void run() {
                        // 判断是否数据错乱, 因为加载图片的过程过程是异步的, 哪些图片先下载好是不一定的
                        // 有可能一张图片先下了, 但是另一张图片后下载的却比它下载的更快
                        String uri =(String)imageView.getTag();
                        if (TextUtils.equals(url, uri)) {
                            imageView.setImageBitmap(bitmap);
                        } else {
                            Log.w(TAG, "The url associated with imageView has changed");
                        }
                    }
                });
            }
        };
        mThreadPoolExecutor.execute(loadBitmapRunnable);
}

这里需要注意的地方是需要将url设置为ImageView的tag, 主要作用是为了防止listView或者recyclerView加载图片错乱的问题。产生错乱的原因是listview复用了机制和异步加载任务造成的。具体可以见这两篇文章 android listview 异步加载图片并防止错位
Android ListView异步加载图片乱序问题,原因分析及解决方案
想起之前面试的时候还被问到怎么解决ListView图片错乱的问题,我直接说我没遇到过这个问题(==...我真没遇到过),现在想来,是我使用的那些图片加载框架已经帮我处理了这个问题。


MemoryCache

MemoryCache这里使用了Android自带的LruCache,全部代码也就这么多

public class MemoryCache implements ImageCache {
    private LruCache<String, Bitmap> mMemoryCache;
    
    public MemoryCache() {
        final int MAX_MEMORY = (int) (Runtime.getRuntime().maxMemory() / 1024);
        final int CACHE_SIZE = MAX_MEMORY / 4;
        mMemoryCache = new LruCache<String, Bitmap>(CACHE_SIZE) {
            @Override
            protected int sizeOf(String key, Bitmap bitmap) {
                return bitmap.getByteCount() / 1024;
            }
        };
    }
    
    @Override
    public Bitmap get(String url) {
        return mMemoryCache.get(getKey(url));
    }
    
    public void put(String url, Bitmap bitmap) {
        mMemoryCache.put(getKey(url), bitmap);
    }
    
    private String getKey(String url) {
        return MD5Util.hashKeyFromUrl(url);
    }
    
    @Override
    public void remove(String url) {
        mMemoryCache.remove(getKey(url));
    }
}

这里就不讲LruCache的原理了,有兴趣可以看这篇彻底解析Android缓存机制——LruCache


DiskCache

DiskCache这里使用了第三方的DiskLruCache,用法可以参考郭霖大神的这篇Android DiskLruCache完全解析,硬盘缓存的最佳方案
主要的方法如下

public Bitmap get(String url, int reqWidth, int reqHeight) {
        Bitmap bitmap = getFromDiskCache(url, reqWidth, reqHeight);
        if (bitmap != null) {
            return bitmap;
        }
        downloadBitmapToDiskCache(url);
        return getFromDiskCache(url, reqWidth, reqHeight);
    }
    
    public void downloadBitmapToDiskCache(String url) {
        String key = MD5Util.hashKeyFromUrl(url);
        try {
            DiskLruCache.Editor editor = mDiskLruCache.edit(key);
            if (editor == null) {
                return;
            }
            // 由于在创建DiskLruCache的时候valueCount指定为1, 所以这里索引传0就可以了
            OutputStream outputStream = editor.newOutputStream(0);
            if (mImageFetcher.downloadSuccess(url, outputStream)) {
                editor.commit();
            } else {
                editor.abort();
            }
            mDiskLruCache.flush();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    
    public Bitmap getFromDiskCache(String url, int reqWidth, int reqHeight) {
        String key = MD5Util.hashKeyFromUrl(url);
        try {
            DiskLruCache.Snapshot snapshot = mDiskLruCache.get(key);
            if (snapshot == null) {
                return null;
            }
            FileInputStream fis = (FileInputStream) snapshot.getInputStream(0);
            if (reqWidth <= 0 || reqHeight <= 0) {
                return BitmapFactory.decodeStream(fis);
            } else {
                return BitmapUtils.getSmallBitmap(fis.getFD(), reqWidth, reqHeight);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

ImageFetcher

ImageFetcher其实是一个接口类,里面包含了一个下载图片到输出流的方法

boolean downloadSuccess(String urlString, OutputStream outputStream);

然后我定义了一个UrlConnectionImageFetcher实现了这个接口,重写了downloadSuccess(url, out)方法

@Override
    public boolean downloadSuccess(String urlString, OutputStream outputStream) {
        if (Looper.myLooper() == Looper.getMainLooper()) {
            throw new RuntimeException("Do not load Bitmap in main thread.");
        }
        HttpURLConnection urlConnection = null;
        BufferedOutputStream out = null;
        BufferedInputStream in = null;
        
        try {
            final URL url = new URL(urlString);
            urlConnection = (HttpURLConnection) url.openConnection();
            in = new BufferedInputStream(urlConnection.getInputStream());
            out = new BufferedOutputStream(outputStream);
            
            int len;
            byte[] buffer = new byte[8 * 1024];
            while ((len = in.read(buffer)) != -1) {
                out.write(buffer, 0, len);
            }
            return true;
        } catch (final Exception e) {
            Log.e(TAG, "Error in downloadBitmap - " + e);
        } finally {
            if (urlConnection != null) {
                urlConnection.disconnect();
            }
            CloseUtil.CloseQuietly(in);
            CloseUtil.CloseQuietly(out);
        }
        return false;
    }

为什么我要这样做呢?其实是因为依赖倒置原则。假如DiskCache直接依赖于UrlConnectionImageFetcher,当我想使用别的网络框架比如OkHttp而不是HttpURLConnection来下载图片时,就必须修改DiskCache的代码。而如果依赖抽象,假如我想用OkHttp来下载,只要再定义一个OkHttpImageFetcher实现ImageFetcher的接口,然后调用DiskCache的setImageFetcher(ImageFetcher fetcher)方法替换其中的ImageFetcher,DiskCache还是调用的ImageFetcher的downloadSuccess()方法,但是调用的却是OkHttpImageFetcher中的具体代码了。


结语

最后上一个github地址
https://github.com/mundane799699/SimpleImageLoader

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

推荐阅读更多精彩内容