西木 2016-10-28
Picasso这个图片框架默认实现了内存中的LRU缓存,但是没有默认实现磁盘缓存(关于磁盘缓存的配置可以看我之前写的一篇博客),我在使用Picasso替换原来的xUtils框架的时候发现内存开销要比之前高好多,于是着手分析Picasso的LRU缓存策略,代码比较好读,下面简单的分析一下。
Picasso加载一个图片的流程一般是这样的:
url->检查LRU缓存中有没有对应的bitmap->调用HTTP框架准备下载该图片资源->http框架检查有没有磁盘缓存->http框架访问网络下载数据并进行缓存
这里面的动作主要是由一个叫BitmapHunter的类完成的。
Picasso有一个接口叫Cache,有一个实现叫LruCache,这个实现类里面是用一个LinkedHashMap<String, Bitmap>来进行缓存,key是图片url,value是bitmap,并不是其他框架爱用的WeakReference方案。
这个实现类里面有几个控制内存使用量的成员,如下:
private final int maxSize;//最大堆内存占用,单位字节
private int size;//当前已经缓存到堆内存中所有bitmap所占的字节数
private int putCount;//将bitmap存入LRU缓存的总次数
private int evictionCount;//因为内存不足而将bitmap移出LRU缓存的总次数
private int hitCount;//从LRU缓存中读取bitmap的总次数
private int missCount;//没有从LRU缓存中根据url找到相应的bitmap的总次数
来看一下添加一个bitmap到缓存的代码
@Override public void set(String key, Bitmap bitmap) {
if (key == null || bitmap == null) {
throw new NullPointerException("key == null || bitmap == null");
}
Bitmap previous;
synchronized (this) {//每次只能读写一个bitmap,因为LinkedHashMap是非线程安全的
putCount++;//存bitmap计数器加一
size += Utils.getBitmapBytes(bitmap);//获取一个bitmap所占内存的字节数
previous = map.put(key, bitmap);//将bitmap存入到hashmap中去,以url为key,如果previous为空说明之前没有存储过该url,否则之前存储过
if (previous != null) {//如果之前已经存储过这个url了
size -= Utils.getBitmapBytes(previous);
}
}
<span style="white-space:pre"> </span>
trimToSize(maxSize);//看看内存占用是否过大,如果太大的话就从LRU缓存中移出一部分bitmap
}
最重要的方法就是这个trimToSize(),它是用来回收bitmap缓存的,让我们来着重研究一下
private void trimToSize(int maxSize) {
while (true) {//一直执行销毁动作,直到当前占用的内存字节数小于规定的最大占用量
String key;
Bitmap value;
synchronized (this) {//由于LinkedHashMap线程非安全,并且只有逐个释放才能准确比较剩余LRU大小,所以要同步执行
if (size < 0 || (map.isEmpty() && size != 0)) {
throw new IllegalStateException(
getClass().getName() + ".sizeOf() is reporting inconsistent results!");
}
if (size <= maxSize || map.isEmpty()) {
break;
}
<span style="white-space:pre"> </span>//LinkedHashMap可以看作是一个先入先出的栈,回收内存的时候先从栈底开始回收,也就是回收好久没用过的bitmap
Map.Entry<String, Bitmap> toEvict = map.entrySet().iterator().next();
key = toEvict.getKey();
value = toEvict.getValue();
map.remove(key);//将bitmap移出LRU缓存
size -= Utils.getBitmapBytes(value);//将当前总堆内存占用量计数器减去移出的bitmap大小
evictionCount++;//回收计数器加一
}
}
}
这个LRU缓存的最核心方法就这样分析完了,其实原理很简单,就是每放一个bitmap进LRU缓存都会记一下这个bitmap的大小,并计算当前LRU的总大小,如果发现总大小太大,就从栈底一个一个的把长时间没用的bitmap给回收掉
那么Picasso如何规定最大内存占用量的呢,让我们来看代码
/** Create a cache using an appropriate portion of the available RAM as the maximum size. */
public LruCache(Context context) {
this(Utils.calculateMemoryCacheSize(context));
}这个LRU缓存类在构造的时候就规定了最大内存占用指标,关键就是这个Utils.calculateMemoryCacheSize()方法,我们来看看它是怎么规定的
static int calculateMemoryCacheSize(Context context) {
ActivityManager am = getService(context, ACTIVITY_SERVICE);
boolean largeHeap = (context.getApplicationInfo().flags & FLAG_LARGE_HEAP) != 0;
int memoryClass = am.getMemoryClass();
if (largeHeap && SDK_INT >= HONEYCOMB) {
memoryClass = ActivityManagerHoneycomb.getLargeMemoryClass(am);
}
// Target ~15% of the available heap.
return 1024 * 1024 * memoryClass / 7;
} 这里是使用Context获得一个ActitityManager,然后用其获得获得一个以MB为单位的APP可占最大堆内存占用大小,然后使用这个最大APP占用的七分之一来做当前图片LRU缓存的最大可用大小,这个最大可用大小当然会随着手机配置的提高而变大,目前我这边测得的数据是:
红米note3: 19MB(3G内存)
Sony L50:22MB(3G内存)
红米2A:17MB(2G内存)
以经验来看,这样的内存分配并不大,经常出现在一个listView或者RecyclerView中,滑到底部后再滑回来,上面的元素的bitmap已经没有的情况,以RGB_8888为例,一个像素占用的大小为32字节,那么一个1920*1080的桌面背景图片所占得堆内存大小是1920*1080*32 = 63MB,对于这样图,LRU几乎是不会缓存的。
关于销毁指定LRU缓存:
手动销毁Picasso提供的默认LRU实现只能做到根据图片url进行销毁,而不能根据某个Activity或者Fragment进行销毁,如果想实现按照页面销毁的话,需要自己重写这个LruCache的实现。下面来看一下根据url进行销毁的源码:
@Override public final synchronized void clearKeyUri(String uri) {
boolean sizeChanged = false;
int uriLength = uri.length();
for (Iterator<Map.Entry<String, Bitmap>> i = map.entrySet().iterator(); i.hasNext();) {
Map.Entry<String, Bitmap> entry = i.next();
String key = entry.getKey();
Bitmap value = entry.getValue();
int newlineIndex = key.indexOf(KEY_SEPARATOR);
if (newlineIndex == uriLength && key.substring(0, newlineIndex).equals(uri)) {//加快寻找速度
i.remove();//将相应的url所对bitmap移出LRU缓存
size -= Utils.getBitmapBytes(value);//将当前总堆内存占用计数器减小被移出的bitmap大小
sizeChanged = true;
}
}
if (sizeChanged) {
trimToSize(maxSize);//移出后执行以下内存检查,如果还是过大就继续销毁栈底的bitmap
}
}
我们在这里发现,Picasso默认的LRU缓存方案并不是我们需要的或者适合自己项目的方案,最好的方法是根据自己APP特点和业务需要重写LruCache,然后换掉Picasso默认的实现方案,方法如下:
Picasso.Builder builder = new Picasso.Builder(getContext());
builder.memoryCache(new CustomLruCache());//设置自定义的缓存方案
Picasso mPicasso = builder.build();//注意自定义Picasso实例要做成全局单例静态,否则缓存会失效同样方法可以自定义下载器,拦截器,线程池等等功能。
分析完这些实现,我们发现Picasso的强大之处并不在于针对某些应用场景提供完美的解决方案,而是它提供了一套完善的接口,让我们自由的根据自己APP的实际情况去自定义我们自己的策略,要想用好Picasso,光用的默认的方法是不行的,更重要的是了解图片下载、缓存、呈现的一系列需求并自定义自己的方案,然后借助Picasso来加载咱们自己的设定。