【发布时间】:2014-10-25 08:22:46
【问题描述】:
我使用Volley NetworkImageView 从互联网下载图像并显示在我的listview 中。现在我想让Volley NetworkImageView 在没有可用网络时显示保存的图像。 Volley 已经缓存了 URL 的图像作为键,因为当我使用
Entry entry = SingletonRequestQueue.getInstance(context).getRequestQueue().getCache().get(imageURL);
entry.data 不为空。但我的问题是图像分辨率很高,我无法使用
Bitmap b = BitmapFactory.decodeByteArray(entry.data, 0, entry.data.length);
因为它会产生很多延迟,我必须重新发明轮子,因为我必须再次创建 asynctask 看看 listview 何时滚动取消解码,回收 bitmap,在内存缓存中创建,找到最佳 insample价值和...
更好的想法是做一些技巧,让Volley NetworkImageView 使用自己的 DiskLRUCache 在没有网络的情况下显示它们。
有什么想法吗?
我的代码:
public class SingletonRequestQueue {
private static SingletonRequestQueue mInstance;
private RequestQueue mRequestQueue;
private ImageLoader mImageLoader;
private static Context mCtx;
private LruBitmapCache mLruBitmapCache;
private SingletonRequestQueue(Context context) {
mCtx = context;
mRequestQueue = getRequestQueue();
mLruBitmapCache = new LruBitmapCache(LruBitmapCache.getCacheSize(context));
mImageLoader = new ImageLoader(mRequestQueue,mLruBitmapCache);
}
public static synchronized SingletonRequestQueue getInstance(Context context) {
if (mInstance == null) {
mInstance = new SingletonRequestQueue(context);
}
return mInstance;
}
public RequestQueue getRequestQueue() {
if (mRequestQueue == null) {
// getApplicationContext() is key, it keeps you from leaking the
// Activity or BroadcastReceiver if someone passes one in.
mRequestQueue = Volley.newRequestQueue(mCtx.getApplicationContext(),new OkHttpStack());
// mRequestQueue = Volley.newRequestQueue(mCtx.getApplicationContext());
}
return mRequestQueue;
}
public <T> void addToRequestQueue(Request<T> req) {
getRequestQueue().add(req);
}
public ImageLoader getImageLoader() {
return mImageLoader;
}
public LruBitmapCache getLruBitmapCache() {
return mLruBitmapCache;
}
public void setLruBitmapCache(LruBitmapCache lruBitmapCache) {
mLruBitmapCache = lruBitmapCache;
}
}
在我的适配器中:
public IssueListAdapter(Context context, int resource, List<Issue> objects) {
super(context, resource, objects);
this.context = context;
this.mIssueList = objects;
mImageLoader = SingletonRequestQueue.getInstance(context).getImageLoader();
}
public static class ViewHolder{
public NetworkImageView mNetworkImageView;
public TextView mFee;
public TextView mName;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if(convertView == null){
holder = new ViewHolder();
LayoutInflater inflater =
(LayoutInflater) context.getSystemService(Activity.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.gridview_issuelist_item, parent, false);
holder.mNetworkImageView = (NetworkImageView)convertView.findViewById(R.id.NetworkImageView_MainActivity_issue_image);
holder.mName = (TextView)convertView.findViewById(R.id.TextView_MainActivity_name);
holder.mFee = (TextView)convertView.findViewById(R.id.TextView_MainActivity_fee);
Utility.settingTypfaceFont(context, holder.mName);
Utility.settingTypfaceFont(context, holder.mFee);
convertView.setTag(holder);
}else{
holder = (ViewHolder)(convertView.getTag());
}
final Issue issue = mIssueList.get(position);
holder.mName.setText(issue.getTitle());
holder.mFee.setText(String.valueOf(issue.getFee()));
String imageURL = issue.getPublicCover();
holder.mNetworkImageView.setImageUrl(imageURL, mImageLoader);
holder.mNetworkImageView.setDefaultImageResId(R.drawable.placeholder2);;
/*
Entry entry = SingletonRequestQueue.getInstance(context).getRequestQueue().getCache().get(imageURL);
if(entry != null && entry.data != null){
byte[] imageByte = entry.data;
loadBitmap(imageByte, holder.mNetworkImageView,imageURL);
}else{
holder.mNetworkImageView.setImageUrl(imageURL, mImageLoader);
}*/
return convertView;
}
@Override
public int getCount() {
if(mIssueList != null){
return mIssueList.size();
}
else{
return 0;
}
}
public List<Issue> getIssueList() {
return mIssueList;
}
}
【问题讨论】:
-
NetworkImageView确实使用了 2 级缓存:开发人员创建的内存缓存和 Volley 自己创建的磁盘缓存。图像首先在内存缓存中检查,然后在磁盘缓存中检查,如果未找到则仅放置网络请求。因此,无需进行任何额外操作即可使其在离线模式下工作。能否分享一下你的代码,看看有没有问题? -
@ManishMulimani 非常感谢您的帮助,我已经更新了我的问题。问题是当我关闭我的应用程序并再次打开它时,只显示图像的占位符,图像下方的文本全部正确加载,但图像没有。再次感谢。
-
ImageLoader 已经有
isCached(String requestUrl, int maxWidth, int maxHeight),如果那里检查的缓存和getRequestQueue().getCache() 不一样,那么重新实现ImageLoader添加一个Context成员,并使用第一个检查缓存的图像在你的问题中。 src代码ImageLoader:android.googlesource.com/platform/frameworks/volley/+/master/… -
DiskBasedCache默认最大缓存大小为 5MB。当您使用非常高分辨率的图像(3000x2500 即 7.5MB)时,您可以增加缓存大小。使用DiskBasedCache(File rootDirectory, int maxCacheSizeInBytes)构造函数指定磁盘大小,即DiskBasedCache(cacheDir, 20 * 1024 * 1024)用于 20MB 缓存。让我知道它是否有帮助。 -
@ManishMulimani 我已经将它设置为
/** Default maximum disk usage in bytes.*/private static final int DEFAULT_DISK_USAGE_BYTES = 20 * 1024 * 1024;