Android高效加载大图、多图解决方案,有效避免程序OOM转载学习研究总结

本文主要是介绍Android高效加载大图、多图解决方案,有效避免程序OOM转载学习研究总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文转载自 郭霖老师的Android高效加载大图、多图解决方案,有效避免程序OOM;

最近研究了郭霖老师的这篇Android高效加载大图的文章,从中学到了很多,也思考了很多。特写此文章将自己的所想结合前辈的文章一起来个总结;


郭老师一共用了两篇文章来介绍android高效加载大图,我在学习了两篇文章之后,将两篇文章结合在了一起,写了一个能自定义压缩图片的高效加载大图多图的Demo,

在参照前辈文章编写的时候,遇到一个关键的技术总结点:

  1. 在通过httpurlconnection从网络获取到输入流之后,一开始我是这样写的:
    public static Bitmap decodeSampledBitmapFromResource(InputStream is, int reqWidth, int reqHeight)
    {
    // 第一次解析将inJustDecodeBounds设置为true,来获取图片大小
    final BitmapFactory.Options options = new BitmapFactory.Options();
    //inJustDecodeBounds设置为true,将不返回实际的bitmap不给其分配内存空间而里面只包括一些解码边界信息即图片大小信息
    options.inJustDecodeBounds = true;
    BitmapFactory.decodeStream(is, null, options);
    // 调用上面定义的方法计算inSampleSize值
    options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
    // 使用获取到的inSampleSize值再次解析图片
    options.inJustDecodeBounds = false;
    return BitmapFactory.decodeStream(is, null, options);
    }
    

        结果执行这个方法之后发现,无论怎么试,方法体返回的Bitmap对象都是null,最后通过debug模式,发现is流在第一次BitmapFactory.decodeStream(is, null, options)之后

就失效了,导致最后return时再调用解析时,已经不存在了。

        后来研究了BitmapFactory的方法,发现里面有一个decodeByteArray()方法,于是尝试了先将is流先转换成byte[]字节数组存放在方法内存中,这样就不会有失效的问题了,于是又了下面改进版的Util.java类:

public class Util
{
/**
* @param is
* @param reqHeight
* @param reqWidth
* @return  
* @throws IOException 
* @Description:压缩图片
*/
public static Bitmap getNewBitmap(InputStream is, int reqHeight, int reqWidth) throws IOException
{
BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
byte[] bt = getBytes(is);
BitmapFactory.decodeByteArray(bt, 0, bt.length, options);
options.inSampleSize = getSampleSize(options, reqHeight, reqWidth);
options.inJustDecodeBounds = false;
return BitmapFactory.decodeByteArray(bt, 0, bt.length, options);
}
/**
* @param options
* @param reqHeight
* @param reqWidth
* @return  
* @Description:获取图片压缩的比率
*/
private static int getSampleSize(BitmapFactory.Options options, int reqHeight, int reqWidth)
{
int height = options.outHeight;
int width = options.outWidth;
int sampleSize = 1;
if(height > reqHeight || width > reqWidth)
{
int heightRatio = Math.round((float) height / (float) reqHeight);
int widthRatio = Math.round((float) width / (float) reqWidth);
sampleSize = heightRatio > widthRatio ? widthRatio : heightRatio;
}
Log.d("gu", sampleSize + "");
return sampleSize;
}
/**
* @param is
* @return
* @throws IOException  
* @Description:将inputStream转换成byte[]
*/
private static byte[] getBytes(InputStream is) throws IOException
{
ByteArrayOutputStream baos = new ByteArrayOutputStream();
int len = -1;
byte[] buffer = new byte[1024];
while((len = is.read(buffer)) != -1)
{
baos.write(buffer, 0, len);
}
baos.close();
return baos.toByteArray();
}
}
经过检验,果然解决了上述问题,并且成功的可以为每张从网络获取到的大图进行不同比列的动态压缩;

下面贴上我的Adapter:
public class MyAdapter extends ArrayAdapterimplements OnScrollListener
{
private GridView myGridView;
private SetmyAsyncTasks;
private LruCachememoryCache;
private int firstVisibleItem;
private int visibleItemCount;
// 记录是不是第一次进入应用,第一次不会激发onScrollStateChanged方法,所以要在onScroll方法中调用下载图片的方法,但是后面就不需要再onScroll中
// 调用下载图片方法了
private boolean isFirstEnter = true;
public MyAdapter(Context context, int resource, String[] objects, GridView gridView)
{
super(context, resource, objects);
myGridView = gridView;
myAsyncTasks = new HashSet();
// 获取应用的最大运行内存
int maxMemory = (int) Runtime.getRuntime().maxMemory();
// 计算分配给LruCache的最大内存
int cacheSize = maxMemory / 8;
// 设置分配给LruCache的内存为应用运行最大内存的8分之一
memoryCache = new LruCache(cacheSize) { @Override protected int sizeOf(String key, Bitmap bitmap) { // 重写此方法来衡量每张图片的大小,默认返回图片数量。 return bitmap.getRowBytes() * bitmap.getHeight() / 1024; } @Override protected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue) { Log.v("tag", "hard cache is full , push to soft cache"); } }; myGridView.setOnScrollListener(this); } @Override public View getView(int position, View convertView, ViewGroup parent) { String url = getItem(position); // ViewHolder holder = null; View view = null; if(convertView == null) { // holder = new ViewHolder(); view = LayoutInflater.from(getContext()).inflate(R.layout.item_layout, null); // convertView.setTag(holder); } else { // holder = (ViewHolder) convertView.getTag(); view = convertView; } ImageView imageView = (ImageView) view.findViewById(R.id.img); // 为了防止异步任务导致图片加载之后出现顺序错乱的情况,为每一个imageView加入一个tag imageView.setTag(url); setImageView(url, imageView); return view; } /** * @param url * @param imageView * @Description:设置图片,如果从缓存中读取到了图片就为它设置为读取到的网络图片,否则将其设置为默认图片 */ private void setImageView(String url, ImageView imageView) { Bitmap bitmap = getBitmapFromCache(url); if(bitmap != null) { imageView.setImageBitmap(bitmap); } else { imageView.setImageResource(R.drawable.ic_launcher); } } /** * @param key * @param bitmap * @Description:如果缓存中还没有这个图片,就将这个图片加入到缓存当中 */ private void addBitmapToCache(String key, Bitmap bitmap) { if(getBitmapFromCache(key) == null) { memoryCache.put(key, bitmap); } } /** * @param key * @return * @Description:根据url这个key从缓存中取出图片 */ private Bitmap getBitmapFromCache(String key) { return memoryCache.get(key); } /** *@Description: imageView的缓存类 *@Author:Nate Robinson *@Since:2015-2-11 */ // class ViewHolder // { // ImageView imageView; // } /** *@Description: 进行下载图片的一步任务类 *@Author:Nate Robinson *@Since:2015-2-11 */ class MyAsyncTask extends AsyncTask { private String imageUrl; @Override protected Bitmap doInBackground(String... params) { imageUrl = params[0]; Bitmap bitmap = downloadBitmap(imageUrl); if(bitmap != null) { addBitmapToCache(imageUrl, bitmap); } return bitmap; } @Override protected void onPreExecute() { Log.d("task", "start"); } @Override protected void onPostExecute(Bitmap result) { Log.d("task", "finish"); super.onPostExecute(result); ImageView imageView = (ImageView) myGridView.findViewWithTag(imageUrl); if(result != null && imageView != null) { imageView.setImageBitmap(result); } myAsyncTasks.remove(this); } /** * @return * @Description:下载任务 */ private Bitmap downloadBitmap(String url) { Bitmap bitmap = null; HttpURLConnection httpURLConnection = null; try { URL imageUrl = new URL(url); httpURLConnection = (HttpURLConnection) imageUrl.openConnection(); httpURLConnection.setConnectTimeout(5 * 1000); httpURLConnection.setReadTimeout(10 * 1000); bitmap = Util.getNewBitmap(httpURLConnection.getInputStream(), 90, 90); } catch(Exception e) { e.printStackTrace(); } finally { // 关闭连接 httpURLConnection.disconnect(); } return bitmap; } } /** * @Description:结束所有在进行中异步任务 */ public void cancelAllTask() { if(myAsyncTasks != null) { for(MyAsyncTask task : myAsyncTasks) { task.cancel(false); } } } private void loadBitmaps(int firstVisibleItem, int visibleItemCount) { try { for(int i = firstVisibleItem; i < firstVisibleItem + visibleItemCount; i++) { String url = com.gu.demo.Images.imageUrls[i]; Bitmap bitmap = getBitmapFromCache(url); if(bitmap == null) { MyAsyncTask task = new MyAsyncTask(); task.execute(url); myAsyncTasks.add(task); } else { // 通过之前的tag从缓存中再次取出ImageView对象 ImageView imageView = (ImageView) myGridView.findViewWithTag(url); if(bitmap != null && imageView != null) { imageView.setImageBitmap(bitmap); } } } } catch(Exception e) { e.printStackTrace(); } } @Override public void onScrollStateChanged(AbsListView view, int scrollState) { // 仅当gridVIew处于静止的时候才去下载图片 if(scrollState == SCROLL_STATE_IDLE) { loadBitmaps(firstVisibleItem, visibleItemCount); } else { // 取消所有的下载任务 cancelAllTask(); } } @Override public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) { this.firstVisibleItem = firstVisibleItem; this.visibleItemCount = visibleItemCount; if(isFirstEnter && visibleItemCount > 0) { loadBitmaps(firstVisibleItem, visibleItemCount); isFirstEnter = false; } } } 
其余代码都是参照前辈的。
写这篇文章的目的,一是给自己一个总结,二是将前辈两篇文章的核心思想结合在一个Demo中,这样更能形象直接的理解学习

这篇关于Android高效加载大图、多图解决方案,有效避免程序OOM转载学习研究总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1068130

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

高效录音转文字:2024年四大工具精选!

在快节奏的工作生活中,能够快速将录音转换成文字是一项非常实用的能力。特别是在需要记录会议纪要、讲座内容或者是采访素材的时候,一款优秀的在线录音转文字工具能派上大用场。以下推荐几个好用的录音转文字工具! 365在线转文字 直达链接:https://www.pdf365.cn/ 365在线转文字是一款提供在线录音转文字服务的工具,它以其高效、便捷的特点受到用户的青睐。用户无需下载安装任何软件,只

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]