Volley DiskBasedCache抛出FileNotFoundException

时间:2014-10-04 13:54:52

标签: android android-volley android-lru-cache

我在Android应用中使用BitmapLRUCache by Trey Robinson进行图片缓存。它是Volley的LRU缓存实现,因为它本身不提供任何图像缓存。

虽然它确实使用DiskBasedCache来缓存HTTP请求。现在出现问题,当DiskBasedCache尝试获取或删除缓存条目时,我反复获取FileNotFoundExceptions。

以下示例日志。

 23833                 Volley  D  [47291] DiskBasedCache.remove: Could not delete cache entry for key=http://a2.mzstatic.com/us/r30/Music1/v4/69/66/0b/69660b50-7771-a43a-919f-26d8b6ae37aa/UMG_cvrart_00602537957941_01_RGB72_1500x1500_14UMGIM31675.400x400-75.jpg, filename=1509125231-2004731303
 23833                 Volley  D  [47291] DiskBasedCache.get: /data/data/com.vibin.billy.debug/cache/volley/6408634861932551223: java.io.FileNotFoundException: /data/data/com.vibin.billy.debug/cache/volley/6408634861932551223: open failed: ENOENT (No such file or directory)
 23833                 Volley  D  [47291] DiskBasedCache.remove: Could not delete cache entry for key=http://a2.mzstatic.com/us/r30/Music4/v4/99/f7/ac/99f7ac13-0dd6-8841-96e0-2a1c18041d84/UMG_cvrart_00602537854097_01_RGB72_1800x1800_14UMGIM03851.400x400-75.jpg, filename=6408634861932551223

为什么DiskBasedCache在使用BitmapLRUcache初始化ImageLoader时会处理图像缓存(见下文)?

ImageLoader imageLoader = new ImageLoader(Volley.newRequestQueue(this), new BitmapLruCache());

以下是我用于缓存的代码。

package com.vibin.billy;

import android.graphics.Bitmap;
import android.support.v4.util.LruCache;
import android.util.Log;

import com.android.volley.toolbox.DiskBasedCache;
import com.android.volley.toolbox.ImageLoader;

/**
 * Basic LRU Memory cache.
 *
 * @author Trey Robinson
 */
public class BitmapLruCache
        extends LruCache<String, Bitmap>
        implements ImageLoader.ImageCache {

    private static final String TAG = BitmapLruCache.class.getSimpleName();

    public BitmapLruCache() {
        this(getDefaultLruCacheSize());
    }

    public BitmapLruCache(int sizeInKiloBytes) {
        super(sizeInKiloBytes);
    }


    @Override
    protected int sizeOf(String key, Bitmap value) {
        return value.getRowBytes() * value.getHeight() / 1024;
    }

    @Override
    public Bitmap getBitmap(String url) {
        //Log.d(TAG, "Grab "+url);
        return get(url);
    }

    @Override
    public void putBitmap(String url, Bitmap bitmap) {
         //Log.d(TAG, "Put "+url);
         put(url, bitmap);
    }

    public static int getDefaultLruCacheSize() {
        final int maxMemory =
                (int) (Runtime.getRuntime().maxMemory() / 1024);
        final int cacheSize = maxMemory / 8;

        Log.d(TAG, "cachesize is " + cacheSize);

        Log.d(TAG,cacheSize+" is cache Size");

        return cacheSize;
    }
}

1 个答案:

答案 0 :(得分:2)

  

为什么DiskBasedCache在我初始化时会处理图像缓存   带有BitmapLRUcache的ImageLoader(见下文)?

用于图像缓存排序使用2级缓存机制,这意味着一个级别位于RAM BitmapLRUcache中,另一个级别位于您的磁盘DiskBasedCache上。为什么?因为从磁盘读取和写入图像需要更长的时间,而不仅仅是简单地读取和写入一些Strings并且性能较差。因此,当您第一次请求Volley下载图片时Volley首先查看缓存级别1,如果您的图片不在那里Volley查看缓存级别2以及您的图片是不是Volley将您的下载请求发送到服务器。

  

当DiskBasedCache尝试时,我反复获取FileNotFoundExceptions   获取或删除缓存条目。

因为您的DiskBasedCache尺寸有限(默认为5MB),而且还是LRU,这意味着Volley想要在DiskBasedCache上存储图片它没有任何空间,它将删除它所持有的一些旧条目并且最近没有引用。(LRU =最近最少使用)因此调用remove函数。