Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现

继续接着上一次https://www.cnblogs.com/webor2006/p/12313876.html的缓存进行编写。

Bitmap复用池:

概念:

关于啥是Bitmap复用,这里还是借用博主的这篇文章https://www.jianshu.com/p/97fd67720b34的说明过一下,先来看两张对比图:

复用前:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第1张图片 

可以看到每一张Bitmp在显示时都申请了一块内存,而如果采用复用Bitmap的情况,则会长成这样:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第2张图片

也就是如果采用了复用之后,如果存在能被复用的图片会重复使用该图片的内存。那复用之后除了减少内存的开销之外,还有另一层意义:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第3张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第4张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第5张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第6张图片

具体实现:

在上一次的LruMemoryCache中还有一块待完善的,就是移出的方法还木有写,如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第7张图片

先把它完善一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第8张图片

另外关于这里面的移除还分为主动和被动移除,如果是调了上面这个remove2很显示是主动移除,此时就不应该回调这个回调了:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第9张图片

为啥?因为如果回调了此方法,是需要将这个资源往复用池中添加的,如果是主动要移除的资源就没必要往复用池中放了;而如果是被动移除的像LruCache中已经达到了最大上限了再添加则就会将最少使用的给移除掉,此时移除掉的就需要放到复用池当中,所以这里需要加个标识来判断一下,如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第10张图片

好,接下来则来开始定义咱们的复用池了,先定义相关的接口:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第11张图片

然后定义具体类:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第12张图片

package com.android.glidearcstudy.glide.recycle;

import android.graphics.Bitmap;

import androidx.collection.LruCache;

public class LruBitmapPool extends LruCache implements BitmapPool {

    private final static int MAX_OVER_SIZE_MULTIPLE = 2;

    public LruBitmapPool(int maxSize) {
        super(maxSize);
    }

    /**
     * 将Bitmap放入复用池
     */
    @Override
    public void put(Bitmap bitmap) {
        //TODO
    }

    /**
     * 获得一个可复用的Bitmap
     */
    @Override
    public Bitmap get(int width, int height, Bitmap.Config config) {
        return null;
    }


    @Override
    protected int sizeOf(Integer key, Bitmap value) {
        return value.getAllocationByteCount();
    }

    @Override
    protected void entryRemoved(boolean evicted, Integer key, Bitmap oldValue, Bitmap newValue) {
        //TODO
    }
}

它也是一个LruCache,接下来具体来实现一下,先实现sizeOf(),跟上次的内存缓存一样:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第13张图片

然后再来实现一下存放方法:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第14张图片

接下来实现get方法,这里从复用池中的所有可复用的bitmap中要找到一个近视差不多大小的图才行,如果要显示的图的大小在复用池中找不到,那此时就不能从复用池来拿了,实现如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第15张图片

图中打问号的则是我们接下来要处理的,如何从复用池中来获取跟我们要申请图片大小近视大的呢?当然我们可以用遍历的方法来实现它,但是这里可以借助于不怎么常用的一个Map比较方便的实现,如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第16张图片

其中NavigableMap是个啥东东,看一下官方的解释:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第17张图片

而我们调用的是这个方法:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第18张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第19张图片

好,接下来继续完善这个get()方法:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第20张图片

好,接下来咱们来应用一下复用池:

public class CacheTest implements Resource.OnResourceListener, MemoryCache.ResourceRemoveListener {

    LruMemoryCache lruMemoryCache;
    ActiveResources activeResource;
    BitmapPool bitmapPool;

    public Resource test(Key key) {
        bitmapPool = new LruBitmapPool(10);
        //内存缓存
        lruMemoryCache = new LruMemoryCache(10);
        lruMemoryCache.setResourceRemoveListener(this);
        //活动资源缓存
        activeResource = new ActiveResources(this);

        /**
         * 第一步 从活动资源中查找是否有正在使用的图片
         */
        Resource resource = activeResource.get(key);
        if (null != resource) {
            //当不使用的时候 release
            resource.acquire();
            return resource;
        }
        /**
         * 第二步 从内存缓存中查找
         */
        resource = lruMemoryCache.get(key);
        if (null != resource) {
            //1.为什么从内存缓存移除?
            // 因为lru可能移除此图片 我们也可能recycle掉此图片
            // 如果不移除,则下次使用此图片从活动资源中能找到,但是这个图片可能被recycle掉了
            lruMemoryCache.remove2(key);//从内存缓存中移除
            resource.acquire();
            activeResource.activate(key, resource);//再加入到活动资源缓存中
            return resource;
        }
        return null;
    }

    /**
     * 这个资源没有正在使用了
     * 将其从活动资源移除
     * 重新加入到内存缓存中
     */
    @Override
    public void onResourceReleased(Resource resource) {
        //TODO
    }

    /**
     * 从内存缓存被动移除
     * 此时得放入复用池
     */
    @Override
    public void onResourceRemoved(Resource resource) {
        bitmapPool.put(resource.getBitmap());
    }
}

最后onResourceReleased()代表该资源没有在使用了,此时需要将它加入到内存缓存中,这里的回调还是增加一个key参数,所以修改一下回调方法:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第21张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第22张图片

至此,关于内存缓存和活动资源缓存的代码就写完了。

磁盘缓存:

当内存缓存木有找到我们想要的图片之后,接下来则需要从磁盘缓存开找了,对于glide官方而言,磁盘缓存也是使用三方开源的,看一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第23张图片

这里就不详细看了,代码量也有点大,先将其拷进工程来直接当工具类来使用既可:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第24张图片

大致瞅一下,其实它也是一个LRU算法,只是说是自己实现了一个,而不是继承至LruCache来实现的:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第25张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第26张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第27张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第28张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第29张图片

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第30张图片

也有像LruCache的trimToSize()的实现:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第31张图片 

接下来咱们利用这个三方开源的磁盘缓存类来使用一下,先新建一个接口:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第32张图片

接下来具体实现一下,也是大致的过一下,跟缓存的实现差不多:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第33张图片

package com.android.glidearcstudy.glide.cache;

import android.content.Context;

import androidx.annotation.Nullable;

import com.android.glidearcstudy.glide.Utils;
import com.android.glidearcstudy.glide.disklrucache.DiskLruCache;

import java.io.File;
import java.io.IOException;
import java.security.MessageDigest;

public class DiskLruCacheWrapper implements DiskCache {


    final static int DEFAULT_DISK_CACHE_SIZE = 250 * 1024 * 1024;
    final static String DEFAULT_DISK_CACHE_DIR = "image_manager_disk_cache";

    private MessageDigest MD;
    private DiskLruCache diskLruCache;

    public DiskLruCacheWrapper(Context context) {
        this(new File(context.getCacheDir(), DEFAULT_DISK_CACHE_DIR), DEFAULT_DISK_CACHE_SIZE);
    }

    protected DiskLruCacheWrapper(File directory, long maxSize) {
        try {
            MD = MessageDigest.getInstance("SHA-256");
            //打开一个缓存目录,如果没有则首先创建它,
            // directory:指定数据缓存地址
            // appVersion:APP版本号,当版本号改变时,缓存数据会被清除
            // valueCount:同一个key可以对应多少文件
            // maxSize:最大可以缓存的数据量
            diskLruCache = DiskLruCache.open(directory, 1, 1, maxSize);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public String getKey(Key key) {
        key.updateDiskCacheKey(MD);
        return new String(Utils.sha256BytesToHex(MD.digest()));
    }

    @Nullable
    @Override
    public File get(Key key) {
        String k = getKey(key);
        File result = null;
        try {
            DiskLruCache.Value value = diskLruCache.get(k);
            if (value != null) {
                result = value.getFile(0);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return result;
    }

    @Override
    public void put(Key key, Writer writer) {
        String k = getKey(key);
        try {
            DiskLruCache.Value current = diskLruCache.get(k);
            if (current != null) {
                return;
            }
            DiskLruCache.Editor editor = diskLruCache.edit(k);
            try {
                File file = editor.getFile(0);
                if (writer.write(file)) {
                    editor.commit();
                }
            } finally {
                editor.abortUnlessCommitted();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void delete(Key key) {
        String k = getKey(key);
        try {
            diskLruCache.remove(k);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void clear() {
        try {
            diskLruCache.delete();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            diskLruCache = null;
        }
    }
}

其中用到了一个加密:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第34张图片

其中我们的Key接口需要定义一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第35张图片

然后它具体的实现类有两个:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第36张图片

实现也比较简单:

package com.android.glidearcstudy.glide.load;

import com.android.glidearcstudy.glide.cache.Key;

import java.security.MessageDigest;

public class EngineKey implements Key {

    private final Object model;
    private final int width;
    private final int height;

    public EngineKey(Object model, int width, int height) {
        this.model = model;
        this.width = width;
        this.height = height;
    }

    @Override
    public void updateDiskCacheKey(MessageDigest messageDigest) {
        messageDigest.update(getKeyBytes());
    }

    @Override
    public byte[] getKeyBytes() {
        return toString().getBytes();
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;

        EngineKey engineKey = (EngineKey) o;

        if (width != engineKey.width) return false;
        if (height != engineKey.height) return false;
        return model != null ? model.equals(engineKey.model) : engineKey.model == null;
    }

    @Override
    public int hashCode() {
        int result = model != null ? model.hashCode() : 0;
        result = 31 * result + width;
        result = 31 * result + height;
        return result;
    }

    @Override
    public String toString() {
        return "EngineKey{" +
                "model=" + model +
                ", width=" + width +
                ", height=" + height +
                '}';
    }
}
package com.android.glidearcstudy.glide.load;

import com.android.glidearcstudy.glide.cache.Key;

import java.security.MessageDigest;

public class ObjectKey implements Key {

    private final Object object;

    public ObjectKey(Object object) {
        this.object = object;
    }

    /**
     * 当磁盘缓存的时候 key只能是字符串
     * ObjectKey变成一个字符串
     * 序列化:json
     * 

* 将ObjectKey转变成一个字符串的手段 * * @param md md5/sha1 */ @Override public void updateDiskCacheKey(MessageDigest md) { md.update(getKeyBytes()); } @Override public byte[] getKeyBytes() { return object.toString().getBytes(); } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; ObjectKey objectKey = (ObjectKey) o; return object != null ? object.equals(objectKey.object) : objectKey.object == null; } @Override public int hashCode() { return object != null ? object.hashCode() : 0; } }

关于磁盘缓存先大致过到这。

图片资源加载:

接下来则是图片加载的处理了,图片加载有N种方式:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第37张图片

所以咱们需要将其抽像一下,如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第38张图片

package com.android.glidearcstudy.glide.load.model;


import com.android.glidearcstudy.glide.cache.Key;
import com.android.glidearcstudy.glide.load.model.data.DataFetcher;

/**
 * @param  此泛型表示的是数据的来源
 * @param   此泛型加载成功后的数据类型,比如InputStream、byte[]
 */
public interface ModelLoader {

    class LoadData {
        //缓存的key
        public final Key key;
        //加载数据
        public final DataFetcher fetcher;

        public LoadData(Key key, DataFetcher fetcher) {
            this.key = key;
            this.fetcher = fetcher;
        }
    }

    /**
     * 此Loader是否能够处理对应Model的数据
     */
    boolean handles(Model model);

    /**
     * 创建加载数据
     */
    LoadData buildData(Model model);
}

其中DataFetcher也是一个接口,不同的形式加载是不一样的:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第39张图片

package com.android.glidearcstudy.glide.load.model.data;

/**
 * 负责数据获取
 */
public interface DataFetcher {

    interface DataFetcherCallback {
        /**
         * 数据加载完成
         */
        void onFetcherReady(Data data);

        /**
         * 加载失败
         */
        void onLoadFaled(Exception e);
    }

    void loadData(DataFetcherCallbacksuper Data> callback);

    void cancel();

    Class getDataClass();

}

好,下面来实现一下具体的数据获取的方式,首先是从网络上获取,最终返回是一个InputStream,所以此时实现如下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第40张图片

package com.android.glidearcstudy.glide.load.model.data;

import android.net.Uri;

import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;

/**
 * 网络数据加载
 */
public class HttpUriFetcher implements DataFetcher {

    private final Uri uri;
    private boolean isCanceled;

    public HttpUriFetcher(Uri uri) {
        this.uri = uri;
    }

    @Override
    public void loadData(DataFetcherCallbacksuper InputStream> callback) {
        HttpURLConnection conn = null;
        InputStream is = null;
        try {
            URL url = new URL(uri.toString());
            conn = (HttpURLConnection) url.openConnection();
            conn.connect();
            is = conn.getInputStream();
            int responseCode = conn.getResponseCode();
            if (isCanceled) {
                return;
            }
            if (responseCode == HttpURLConnection.HTTP_OK) {
                callback.onFetcherReady(is);
            } else {
                callback.onLoadFaled(new RuntimeException(conn.getResponseMessage()));
            }
        } catch (Exception e) {
            callback.onLoadFaled(e);
        } finally {
            if (null != is) {
                try {
                    is.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (null != conn) {
                conn.disconnect();
            }
        }
    }

    @Override
    public void cancel() {
        isCanceled = true;
    }

    @Override
    public Class getDataClass() {
        return InputStream.class;
    }
}

而文件数据加载则是:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第41张图片

package com.android.glidearcstudy.glide.load.model.data;

import android.content.ContentResolver;
import android.net.Uri;

import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;

/**
 * 文件数据加载
 */
public class FileUriFetcher implements DataFetcher {

    private final Uri uri;
    private final ContentResolver cr;

    public FileUriFetcher(Uri uri, ContentResolver cr) {
        this.uri = uri;
        this.cr = cr;
    }

    @Override
    public void loadData(DataFetcherCallbacksuper InputStream> callback) {
        InputStream is = null;
        try {
            is = cr.openInputStream(uri);
            callback.onFetcherReady(is);
        } catch (FileNotFoundException e) {
            callback.onLoadFaled(e);
        } finally {
            if (null != is) {
                try {
                    is.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @Override
    public void cancel() {

    }

    @Override
    public Class getDataClass() {
        return InputStream.class;
    }
}

好,对于之前定义的ModelLoader定义了接口还木有定义实现类,下面针对HTTP的来实现一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第42张图片

package com.android.glidearcstudy.glide.load.model;

import android.net.Uri;

import com.android.glidearcstudy.glide.load.ObjectKey;
import com.android.glidearcstudy.glide.load.model.data.HttpUriFetcher;

import java.io.InputStream;

public class HttpUriLoader implements ModelLoader {

    /**
     * http类型的uri此loader才支持,只支持http和https
     */
    @Override
    public boolean handles(Uri uri) {
        String scheme = uri.getScheme();
        return scheme.equalsIgnoreCase("http") || scheme.equalsIgnoreCase("https");
    }

    @Override
    public LoadData buildData(Uri uri) {
        return new LoadData(new ObjectKey(uri), new HttpUriFetcher(uri));
    }

}

可以看到这个ModelLoader就已经将要加载的所有东东都组装好,那怎么用呢?下面用测试类来试一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第43张图片

可见使用非常之灵活,目前只实现了一个Http的加载器,还有其它的加载类型只要对ModelLoader进行拓展既可,基实可以看一下glide官方其实拓展了N多个Loader,可以大致瞅一下:

Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现_第44张图片

重点是要学会这种基础灵活的框架搭建的思路,目前先只创建了一个http的加载器,关于其它的Loader的创建下次再继续。

你可能感兴趣的:(Glide框架设计<二>-------Glide复用池、磁盘缓存、图片资源加载手写实现)