全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

详解Android_性能优化之ViewPager加载成百上千高清大图oom解决方案

一、背景

最近做项目需要用到选择图片上传,类似于微信、微博那样的图片选择器,ContentResolver读取本地图片资源并用RecyclerView+Glide加载图片显示就搞定列表的显示,这个没什么大问题,重点是,点击图片进入大图浏览,比如你相册有几百张图片,也就意味着在ViewPager中需要加载几百个view,况且手机拍出来的图片都是1-2千万左右像素的高清大图(笔者手机2千万像素 也就是拍照出来的照片3888*5152),大小也有5-7个兆,ViewPager滑动不了十几张就oom了,即是对图片做了压缩处理,把图片分辨率降低至1366*960,大小压缩至150k以下,并且在ViewPager的destroyItem方法做了bitmap资源的回收,虽然效果好了点,这也抵挡不了oom的降临(网上查找的方案都是压缩、使用第三方控件、回收,其实这都没用,可能他们没有真正体验过ViewPager加载几百上千张大图的感受),浏览到了第50-70张的时候就oom了 内存一直暴涨,根本回收不了的,不信你们试试,压缩和回收根本不能根治问题,那么怎么解决呢?研究了微信和微博,他们怎么也不会oom,最后我想到了一种解决方案。

二、方案实施

1、以往的普通做法

部分代码:

List<SubsamplingScaleImageView> mViews = new ArrayList<>(); 
     
    int size = mDatas.size(); 
    for (int i = 0; i < size; i++) { 
      SubsamplingScaleImageView view = new SubsamplingScaleImageView(this); 
      mViews.add(view); 
    } 
 
    mBinding.viewpager.setAdapter(new MyAdapter()); 
class MyAdapter extends PagerAdapter { 
 
    @Override 
    public int getCount() { 
      return mDatas.size(); 
    } 
 
    @Override 
    public boolean isViewFromObject(View view, Object object) { 
      return view == object; 
    } 
 
    @Override 
    public Object instantiateItem(ViewGroup container, final int position) { 
 
      ViewGroup.LayoutParams params = new ViewGroup.LayoutParams( 
          ViewPager.LayoutParams.MATCH_PARENT,ViewPager.LayoutParams.MATCH_PARENT); 
      final SubsamplingScaleImageView imageView = mViews.get(position); 
      imageView.setLayoutParams(params); 
 
      final String url = mDatas.get(position); 
      String cacheExists = cacheExists(url); 
      if(TextUtils.isEmpty(cacheExists)) {//没缓存 需要压缩(压缩耗时 异步) 
        new AsyncTask<Void, Void, String>() { 
          @Override 
          protected String doInBackground(Void... voids) { 
            String cacheNoExistsPath = getCacheNoExistsPath(url); 
            BitmapCompressUtils.compressBitmap(url, cacheNoExistsPath); 
            File file = new File(cacheNoExistsPath); 
            if (file.exists()) {//存在表示成功 
              return cacheNoExistsPath; 
            } else { 
              return url; 
            } 
          } 
 
          @Override 
          protected void onPostExecute(String s) { 
            imageView.setImage(ImageSource.uri(s)); 
          } 
 
        }.execute(); 
 
 
      } else {//有缓存 直接显示 
        imageView.setImage(ImageSource.uri(cacheExists)); 
      } 
 
      container.addView(imageView); 
      return imageView; 
 
    } 
 
    @Override 
    public void destroyItem(ViewGroup container, int position, Object object) { 
 
      SubsamplingScaleImageView imageView = mViews.get(position); 
      if(imageView != null) { 
        imageView.recycle(); 
      } 
 
      container.removeView(imageView); 
 
    } 
  } 

/** 
   * 判断当前图片url对应的压缩过的缓存是否存在 ""表示不存在 
   * 
   * @param url 图片路径 
   * @return 
   */ 
  private String cacheExists(String url) { 
    try { 
      File fileDir = new File(mCacheRootPath); 
      if(!fileDir.exists()) { 
        fileDir.mkdirs(); 
      } 
 
      File file = new File(mCacheRootPath,new StringBuffer().append(MD5EncryptorUtils.md5Encryption(url)).toString()); 
      if(file.exists()) { 
        return file.getAbsolutePath(); 
      } 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
 
    return ""; 
  } 
 
  public String getCacheNoExistsPath(String url) { 
    File fileDir = new File(mCacheRootPath); 
    if(!fileDir.exists()) { 
      fileDir.mkdirs(); 
    } 
 
 
    return new StringBuffer().append(mCacheRootPath) 
        .append(MD5EncryptorUtils.md5Encryption(url)).toString(); 
  } 

可以看到,这里笔者通过自己的压缩算法(上一篇文章Android_NDK图片压缩之Libjpeg库使用 )做了图片压缩,并缓存,细心的朋友应该有发现mViews集合添加的view个数是mDatas的size大小个数,这样就会导致一个问题ViewPager一直向下滑动的时候,内存一直是增加的,即是做了资源回收,也是不能解决问题(况且笔者这里展示图片的控件是SubsamplingScaleImageView 很不错的大图局部加载控件 能有效防止oom),大家可以试试,大量图片的时候还是会oom,这得归根于viewpager加载的图片数量问题。

2、解决方案:

图片压缩也做了,资源回收也做了,但是ViewPager加载越来越多图片的时候就会oom 你避免不了,不信你试试;

这里就要用到ViewPager的view的重用机制(自己理解的),也就是mViews我们固定给定个数量,如4,这样ViewPager的i实际所需要的item也就只有4个。

修改后的部分代码:

for (int i = 0; i < 4; i++) { 
      SubsamplingScaleImageView view = new SubsamplingScaleImageView(this); 
      mViews.add(view); 
    } 
 
    mBinding.viewpager.setAdapter(new MyAdapter()); 
class MyAdapter extends PagerAdapter { 
 
    @Override 
    public int getCount() { 
      return mDatas.size(); 
    } 
 
    @Override 
    public boolean isViewFromObject(View view, Object object) { 
      return view == object; 
    } 
 
    @Override 
    public Object instantiateItem(ViewGroup container, final int position) { 
 
      ViewGroup.LayoutParams params = new ViewGroup.LayoutParams( 
          ViewPager.LayoutParams.MATCH_PARENT,ViewPager.LayoutParams.MATCH_PARENT); 
 
      int i = position % 4; 
      final SubsamplingScaleImageView imageView = mViews.get(i); 
      imageView.setLayoutParams(params); 
 
      final String url = mDatas.get(position); 
      String cacheExists = cacheExists(url); 
      if(TextUtils.isEmpty(cacheExists)) {//没缓存 需要压缩(压缩耗时 异步) 
        new AsyncTask<Void, Void, String>() { 
          @Override 
          protected String doInBackground(Void... voids) { 
            String cacheNoExistsPath = getCacheNoExistsPath(url); 
            BitmapCompressUtils.compressBitmap(url, cacheNoExistsPath); 
            File file = new File(cacheNoExistsPath); 
            if (file.exists()) {//存在表示成功 
              return cacheNoExistsPath; 
            } else { 
              return url; 
            } 
          } 
 
          @Override 
          protected void onPostExecute(String s) { 
            imageView.setImage(ImageSource.uri(s)); 
          } 
 
        }.execute(); 
 
 
      } else {//有缓存 直接显示 
        imageView.setImage(ImageSource.uri(cacheExists)); 
      } 
 
      container.addView(imageView); 
      return imageView; 
 
    } 
 
    @Override 
    public void destroyItem(ViewGroup container, int position, Object object) { 
      int i = position % 4; 
      SubsamplingScaleImageView imageView = mViews.get(i); 
      if(imageView != null) { 
        imageView.recycle(); 
      } 
 
      container.removeView(imageView); 
 
    } 

很简单的修改 就能有效防止oom  利用position%4拿到第几个控件从mViews取值,保证了viewpager加载的mViews存储的图片为4个

看一直向下滑动的内存走势图

内存基本维持稳定

三、demo演示

因为要读取相册就没再模拟器运行录制gif ,直接截图


 

demo下载:demo

四、总结

这个只是简单的演示,实际项目中的相册比这个复杂多了,简单说就是要压缩,要回收,View重用。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# viewpager加载图片oom  # android加载图片oom  # viewpager加载大图片  # 详解Android性能优化之内存泄漏  # Android性能优化之利用强大的LeakCanary检测内存泄漏及解决办法  # Android开发性能优化总结  # Android布局性能优化之按需加载View  # 浅析安卓(Android)的性能优化  # Android高级开发之性能优化典范  # Android性能优化以及数据优化方法  # Android中SparseArray性能优化的使用方法  # 详解Android性能优化之启动优化  # 加载  # 都是  # 就会  # 也就  # 不信  # 即是  # 也做  # 自己的  # 我想  # 也有  # 好了  # 就能  # 上一  # 这也  # 很简单  # 不存在  # 可以看到  # 很不错  # 解决问题  # 十几 


相关文章: 高防服务器租用首荐平台,企业级优惠套餐快速部署  c++怎么实现高并发下的无锁队列_c++ std::atomic原子变量与CAS操作【详解】  上海网站制作网站建设公司,建筑电工证网上查询系统入口?  如何用y主机助手快速搭建网站?  ,交易猫的商品怎么发布到网站上去?  广州美橙建站如何快速搭建多端合一网站?  网站建设制作、微信公众号,公明人民医院怎么在网上预约?  定制建站是什么?如何实现个性化需求?  建站之星安装后如何配置SEO及设计样式?  Python路径拼接规范_跨平台处理说明【指导】  手机网站制作与建设方案,手机网站如何建设?  建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略  制作农业网站的软件,比较好的农业网站推荐一下?  详解jQuery中基本的动画方法  建站之星安装后如何自定义网站颜色与字体?  实惠建站价格推荐:2025年高性价比自助建站套餐解析  相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?  建站之星ASP如何实现CMS高效搭建与安全管理?  免费制作小说封面的网站有哪些,怎么接网站批量的封面单?  济南网站建设制作公司,室内设计网站一般都有哪些功能?  已有域名和空间如何快速搭建网站?  建站之星如何优化SEO以实现高效排名?  番禺网站制作公司哪家值得合作,番禺图书馆新馆开放了吗?  宝塔面板创建网站无法访问?如何快速排查修复?  在线ppt制作网站有哪些,请推荐几个好的课件下载的网站?  招商网站制作流程,网站招商广告语?  ,购物网站怎么盈利呢?  如何快速搭建高效WAP手机网站吸引移动用户?  简易网站制作视频教程,使用记事本编写一个简单的网页html文件?  如何用PHP快速搭建高效网站?分步指南  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  如何用虚拟主机快速搭建网站?详细步骤解析  为什么Go需要go mod文件_Go go mod文件作用说明  长春网站建设制作公司,长春的网络公司怎么样主要是能做网站的?  php条件判断怎么写_ifelse和switchcase的使用区别【对比】  太平洋网站制作公司,网络用语太平洋是什么意思?  C#如何使用XPathNavigator高效查询XML  建站上传速度慢?如何优化加速网站加载效率?  Swift中switch语句区间和元组模式匹配  广州顶尖建站服务:企业官网建设与SEO优化一体化方案  建站之星价格显示格式升级,你的预算足够吗?  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  香港服务器如何优化才能显著提升网站加载速度?  阿里云网站搭建费用解析:服务器价格与建站成本优化指南  建站之星安装需要哪些步骤及注意事项?  如何访问已购建站主机并解决登录问题?  如何在Golang中使用replace替换模块_指定本地或远程路径  如何将凡科建站内容保存为本地文件?  大连网站设计制作招聘信息,大连投诉网站有哪些?  陕西网站制作公司有哪些,陕西凌云电器有限公司官网? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。