当前位置: 首页 > news >正文

Android 避免bitmap内存限制

       在编写Android程序的时候,我们总是难免会碰到OOM(OUT OF MEMORY)的错误,那么这个错误究竟是怎么来的呢,可以先看一下这篇文章ANDROID BITMAP内存限制OOM,OUT OF MEMORY。

       这里,我使用Gallery来举例,在模拟器中,不会出现OOM错误,但是,一旦把程序运行到真机里,图片文件一多,必然会出现OOM,我们通过做一些额外的处理来避免。

       1.创建一个图片缓存对象HashMap dataCache,integer对应Adapter中的位置position,我们只用缓存处在显示中的图片,对于之外的位置,如果dataCache中有对应的图片,我们需要进行回收内存。在这个例子中,Adapter对象的getView方法首先判断该位置是否有缓存的bitmap,如果没有,则解码图片(bitmapDecoder.getPhotoItem,BitmapDecoder类见后面)并返回bitmap对象,设置dataCache 在该位置上的bitmap缓存以便之后使用;若是该位置存在缓存,则直接取出来使用,避免了再一次调用底层的解码图像需要的内存开销。有时为了提高 Gallery的更新速度,我们还可以预存储一些位置上的bitmap,比如存储显示区域位置外向上3个向下3个位置的bitmap,这样上或下滚动 Gallery时可以加快getView的获取。

java代码:
  1. public View getView(int position, View convertView, ViewGroup parent) {

  2. if(convertView==null){
  3. LayoutInflater inflater = LayoutInflater.from(context);
  4. convertView = inflater.inflate(R.layout.photo_item, null);
  5. holder = new ViewHolder();
  6. holder.photo = (ImageView) convertView.findViewById(R.id.photo_item_image);
  7. holder.photoTitle = (TextView) convertView.findViewById(R.id.photo_item_title);
  8. holder.photoDate = (TextView) convertView.findViewById(R.id.photo_item_date);
  9. convertView.setTag(holder);
  10. }else {
  11. holder = (ViewHolder) convertView.getTag();
  12. }
  13. cursor.moveToPosition(position);
  14. Bitmap current = dateCache.get(position);
  15. if(current != null){//如果缓存中已解码该图片,则直接返回缓存中的图片
  16. holder.photo.setImageBitmap(current);
  17. }else {
  18. current = bitmapDecoder.getPhotoItem(cursor.getString(1), 2) ;
  19. holder.photo.setImageBitmap(current);
  20. dateCache.put(position, current);
  21. }

  22. holder.photoTitle.setText(cursor.getString(2));
  23. holder.photoDate.setText(cursor.getString(4));
  24. return convertView;
  25. }

  26. }
复制代码

java代码:
  1. package eoe.bestjoy;

  2. import java.io.FileNotFoundException;
  3. import java.io.FileOutputStream;
  4. import android.content.Context;
  5. import android.graphics.Bitmap;
  6. import android.graphics.BitmapFactory;
  7. import android.graphics.Matrix;

  8. public class BitmapDecoder {

  9. private static final String TAG = "BitmapDecoder";
  10. private Context context;
  11. public BitmapDecoder(Context context) {
  12. this.context = context;
  13. }

  14. public Bitmap getPhotoItem(String filepath,int size) {
  15. BitmapFactory.Options options = new BitmapFactory.Options();
  16. options.inSampleSize=size;
  17. Bitmap bitmap = BitmapFactory.decodeFile(filepath,options);
  18. bitmap=Bitmap.createScaledBitmap(bitmap, 180, 251, true);
  19. //预先缩放,避免实时缩放,可以提高更新率
  20. return bitmap;
  21. }

  22. }
复制代码

       2.由于Gallery控件的特点,总有一个item处于当前选择状态,我们利用此时进行dataCache中额外不用的bitmap的清理,来释放内存。

java代码:
  1. @Override
  2. public void onItemSelected(AdapterView<?> parent, View view, int position,long id) {

  3. releaseBitmap();
  4. Log.v(TAG, "select id:"+ id);
  5. }

  6. private void releaseBitmap(){
  7. //在这,我们分别预存储了第一个和最后一个可见位置之外的3个位置的bitmap
  8. //即dataCache中始终只缓存了(M=6+Gallery当前可见view的个数)M个bitmap
  9. int start = mGallery.getFirstVisiblePosition()-3;
  10. int end = mGallery.getLastVisiblePosition()+3;


  11. Log.v(TAG, "start:"+ start);
  12. Log.v(TAG, "end:"+ end);

  13. //释放position<start之外的bitmap资源
  14. Bitmap delBitmap;
  15. for(int del=0;del<start;del++){
  16. delBitmap = dateCache.get(del);
  17. if(delBitmap != null){
  18. //如果非空则表示有缓存的bitmap,需要清理
  19. Log.v(TAG, "release position:"+ del);
  20. //从缓存中移除该del->bitmap的映射
  21. dateCache.remove(del);
  22. delBitmap.recycle();
  23. }
  24. }

  25. freeBitmapFromIndex(end);
  26. }

  27. /**
  28. * 从某一位置开始释放bitmap资源
  29. * @param index
  30. */

  31. private void freeBitmapFromIndex(int end) {
  32. //释放之外的bitmap资源
  33. Bitmap delBitmap;
  34. for(int del =end+1;del<dateCache.size();del++){
  35. delBitmap = dateCache.get(del);
  36. if(delBitmap != null){
  37. dateCache.remove(del);
  38. delBitmap.recycle();
  39. Log.v(TAG, "release position:"+ del);
  40. }

  41. }
  42. }

from:http://www.eoeandroid.com/thread-80104-1-1.html

相关文章:

  • 圆角背景的ListView
  • LayerDrawable层叠样式layer-list
  • android的edittext默认不获得焦点
  • Android中AsyncTask的用法实例
  • Eclipse中如何更改字体及字体大小
  • Android通过http协议POST传输方式
  • Android四种Activity的加载模式
  • 谷歌Android UI设计技巧:优秀UI设计准则
  • android UI优化
  • ListView性能优化之视图缓存
  • Android中ListView的addFooterView不显示的问题
  • iPhone入门学习——Objective-C学习文档
  • Android 开发中的网络下载策略 与 文件缓存策略
  • 关于代码重构的一些思考
  • Android使用VideoView播放网络视频
  • Apache Spark Streaming 使用实例
  • Computed property XXX was assigned to but it has no setter
  • Docker 笔记(2):Dockerfile
  • Essential Studio for ASP.NET Web Forms 2017 v2,新增自定义树形网格工具栏
  • Fundebug计费标准解释:事件数是如何定义的?
  • learning koa2.x
  • leetcode378. Kth Smallest Element in a Sorted Matrix
  • magento 货币换算
  • MySQL QA
  • PaddlePaddle-GitHub的正确打开姿势
  • Redis的resp协议
  • SQLServer插入数据
  • vue 配置sass、scss全局变量
  • 初探 Vue 生命周期和钩子函数
  • 普通函数和构造函数的区别
  • 前端js -- this指向总结。
  • 前端临床手札——文件上传
  • 前端之Sass/Scss实战笔记
  • 树莓派 - 使用须知
  • 为视图添加丝滑的水波纹
  • 延迟脚本的方式
  • 我们雇佣了一只大猴子...
  • ​【C语言】长篇详解,字符系列篇3-----strstr,strtok,strerror字符串函数的使用【图文详解​】
  • # Apache SeaTunnel 究竟是什么?
  • #include<初见C语言之指针(5)>
  • #多叉树深度遍历_结合深度学习的视频编码方法--帧内预测
  • (20)目标检测算法之YOLOv5计算预选框、详解anchor计算
  • (poj1.2.1)1970(筛选法模拟)
  • (ZT)薛涌:谈贫说富
  • (二十一)devops持续集成开发——使用jenkins的Docker Pipeline插件完成docker项目的pipeline流水线发布
  • (附源码)spring boot公选课在线选课系统 毕业设计 142011
  • (十)T检验-第一部分
  • (五)网络优化与超参数选择--九五小庞
  • (一一四)第九章编程练习
  • (转)Oracle存储过程编写经验和优化措施
  • .bat批处理(十一):替换字符串中包含百分号%的子串
  • .Net Core与存储过程(一)
  • .NET MVC 验证码
  • .Net 中Partitioner static与dynamic的性能对比
  • .NET高级面试指南专题十一【 设计模式介绍,为什么要用设计模式】