viewpage listview gridview加载本地大图多图OOM处理办法

时间:2023-03-09 05:11:01
viewpage listview gridview加载本地大图多图OOM处理办法

很少上博客园写东西了.

最近在写公司项目,由于需要加载本地相册通过viewpager方式来加载,

最后发现直接进入界面就OOM了.

经过几天的整理最终搞定.

现在将加载本地和加载网络图片的缓存工具类贴出代码.

需要用的直接调用其中的暴露出的方法,传入本地图片地址或网络图片地址 +imageview即可

代码如有问题请及时跟帖或者加QQ30338970交流.

 package cn.haodehaode.utils;

 import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.support.v4.util.LruCache;
import android.widget.ImageView; import java.net.HttpURLConnection;
import java.net.URL;
import java.util.HashSet;
import java.util.Set; import cn.haodehaode.R; /**
* 图片缓存
* 异步加载网络图片
*
* @author JALEN c9n9m@163.com
* @version V1.0
* @Title: ${FILE_NAME}
* @Package cn.haodehaode.utils
* @Description: ${todo}
* @date 15/10/29 13:39
*/
public class LruCacheImgUrlUtils { private static LruCache<String, Bitmap> mMemoryCache; private static LruCacheImgUrlUtils lruCacheSingleton; /**
* 记录所有正在下载或等待下载的任务。
*/
private static Set<BitmapWorkerTasks> taskCollection; public static synchronized LruCacheImgUrlUtils getLruUrlCacheSingleton() {
if (lruCacheSingleton == null) {
lruCacheSingleton = new LruCacheImgUrlUtils(); taskCollection = new HashSet<BitmapWorkerTasks>();
// LruCache通过构造函数传入缓存值,以KB为单位。
int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
// 使用最大可用内存值的1/8作为缓存的大小。
int cacheSize = maxMemory / 8;
mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {
@Override
protected int sizeOf(String key, Bitmap bitmap) {
// 重写此方法来衡量每张图片的大小,默认返回图片数量。
return bitmap.getByteCount() / 1024;
}
};
}
return lruCacheSingleton;
} /**
* 将一张图片存储到LruCache中。
*
* @param key LruCache的键,这里传入图片的URL地址。
* @param bitmap LruCache的键,这里传入从网络上下载的Bitmap对象。
*/
public void addBitmapToMemoryCache(String key, Bitmap bitmap) {
if (getBitmapFromMemoryCache(key) == null) {
mMemoryCache.put(key, bitmap);
}
} /**
* 从LruCache中获取一张图片,如果不存在就返回null。
*
* @param key LruCache的键,这里传入图片的URL地址。
* @return 对应传入键的Bitmap对象,或者null。
*/
public Bitmap getBitmapFromMemoryCache(String key) {
return mMemoryCache.get(key);
} /**
* 给ImageView设置图片。首先从LruCache中取出图片的缓存,设置到ImageView上。如果LruCache中没有该图片的缓存,
* 就给ImageView设置一张默认图片。
*
* @param imageUrl 图片的URL地址,用于作为LruCache的键。
* @param imageView 用于显示图片的控件。
*/
public void setImageView(String imageUrl, ImageView imageView) {
Bitmap bitmap = getBitmapFromMemoryCache(imageUrl);
if (bitmap != null) {
imageView.setImageBitmap(bitmap);
} else {
imageView.setImageResource(R.drawable.empty_photo);
BitmapWorkerTasks task = new BitmapWorkerTasks(imageView);
task.execute(imageUrl);
}
} /**
* 异步下载图片的任务。
*
* @author guolin
*/
class BitmapWorkerTasks extends AsyncTask<String, Void, Bitmap> { private ImageView mImageView; public BitmapWorkerTasks(ImageView imageView) {
mImageView = imageView;
} /**
* 图片的URL地址
*/
private String imageUrl; @Override
protected Bitmap doInBackground(String... params) {
imageUrl = params[0];
// 在后台开始下载图片
Bitmap bitmap = downloadBitmap(params[0]);
if (bitmap != null) {
// 图片下载完成后缓存到LrcCache中
addBitmapToMemoryCache(params[0], bitmap);
}
return bitmap;
} @Override
protected void onPostExecute(Bitmap bitmap) {
super.onPostExecute(bitmap);
// 根据Tag找到相应的ImageView控件,将下载好的图片显示出来。
if (mImageView != null && bitmap != null) {
mImageView.setImageBitmap(bitmap);
}
taskCollection.remove(this);
} /**
* 建立HTTP请求,并获取Bitmap对象。
*
* @param imageUrl 图片的URL地址
* @return 解析后的Bitmap对象
*/
private Bitmap downloadBitmap(String imageUrl) {
Bitmap bitmap = null;
HttpURLConnection con = null;
try {
URL url = new URL(imageUrl);
con = (HttpURLConnection) url.openConnection();
con.setConnectTimeout(5 * 1000);
con.setReadTimeout(10 * 1000);
bitmap = BitmapFactory.decodeStream(con.getInputStream());
} catch (Exception e) {
e.printStackTrace();
} finally {
if (con != null) {
con.disconnect();
}
}
return bitmap;
} } }
package cn.haodehaode.utils;

import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.support.v4.util.LruCache;
import android.widget.ImageView; import java.net.HttpURLConnection;
import java.net.URL;
import java.util.HashSet;
import java.util.Set; import cn.haodehaode.R; /**
* 图片缓存
*
* @author JALEN c9n9m@163.com
* @version V1.0
* @Title: ${FILE_NAME}
* @Package cn.haodehaode.utils
* @Description: ${todo}
* @date 15/10/29 13:39
*/
public class LruCacheImgUtils { private static LruCache<String, Bitmap> mMemoryCache; private static LruCacheImgUtils lruCacheSingleton; public static synchronized LruCacheImgUtils getLruCacheSingleton() {
if (lruCacheSingleton == null) {
lruCacheSingleton = new LruCacheImgUtils(); // LruCache通过构造函数传入缓存值,以KB为单位。
int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
// 使用最大可用内存值的1/8作为缓存的大小。
int cacheSize = maxMemory / 8;
mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {
@Override
protected int sizeOf(String key, Bitmap bitmap) {
// 重写此方法来衡量每张图片的大小,默认返回图片数量。
return bitmap.getByteCount() / 1024;
}
};
}
return lruCacheSingleton;
} public void addBitmapToMemoryCache(String key, Bitmap bitmap) {
if (getBitmapFromMemCache(key) == null) {
mMemoryCache.put(key, bitmap);
}
} public Bitmap getBitmapFromMemCache(String key) {
return mMemoryCache.get(key);
} public void loadBitmap(String path, ImageView imageView) {
final Bitmap bitmap = getBitmapFromMemCache(path);
if (bitmap != null) {
imageView.setImageBitmap(bitmap);
} else {
imageView.setImageResource(R.drawable.default_image);
BitmapWorkerTask task = new BitmapWorkerTask(imageView);
task.execute(path);
}
} public Bitmap decodeSampledBitmapFromResource(String path,
int reqWidth, int reqHeight) {
// 第一次解析将inJustDecodeBounds设置为true,来获取图片大小
final BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeFile(path, options);
// 调用上面定义的方法计算inSampleSize值
options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
// 使用获取到的inSampleSize值再次解析图片
options.inJustDecodeBounds = false;
return BitmapFactory.decodeFile(path, options);
} public static int calculateInSampleSize(BitmapFactory.Options options,
int reqWidth, int reqHeight) {
// 源图片的高度和宽度
final int height = options.outHeight;
final int width = options.outWidth;
int inSampleSize = 1;
if (height > reqHeight || width > reqWidth) {
// 计算出实际宽高和目标宽高的比率
final int heightRatio = Math.round((float) height / (float) reqHeight);
final int widthRatio = Math.round((float) width / (float) reqWidth);
// 选择宽和高中最小的比率作为inSampleSize的值,这样可以保证最终图片的宽和高
// 一定都会大于等于目标的宽和高。
inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;
}
return inSampleSize;
} class BitmapWorkerTask extends AsyncTask<String, Void, Bitmap> { private ImageView mImageView; public BitmapWorkerTask(ImageView imageView) {
mImageView = imageView;
} // 在后台加载图片。
@Override
protected Bitmap doInBackground(String... params) {
final Bitmap bitmap = decodeSampledBitmapFromResource(
params[0], 320, 320);
addBitmapToMemoryCache(String.valueOf(params[0]), bitmap);
return bitmap;
} @Override
protected void onPostExecute(Bitmap bitmap) {
super.onPostExecute(bitmap);
mImageView.setImageBitmap(bitmap);
}
}
}