天翼云代理,天翼云代理商,北京代理商
天翼云2.5折专线:18510009100/18510009200(全国市话)

【Android 内存优化】自定义组件长图组件 ( 获取图像宽高 | 计算解码区域 | 设置图像解码属性 复用 像素格式 | 图像绘制 )

2022-01-26 03:32:14

作者 | 扬阜

文章目录

一、获取图像真实宽高

二、核算解码区域

三、设置解码参数 内存复用 像素格式

四、图像制造

五、施行效果

六、源码及资源下载



官方文档 API : BitmapRegionDecoder



在【Android 内存优化】自定义组件长图组件 ( 自定义组件结构办法 ) 基础上继续开发 ;






一、获取图像真实宽高


闪现的图像是一张长图 , 在该组件中 , 宽度肯定要无缺闪现出来 , 解码图片的不同高度的数据 ;


首先要测量图片数据的真实宽高 , 然后根据图像的宽高 , 与组件的宽高 , 以及要闪现的图像方位 , 核算要解码的图像区域 ;



参看 【Android 内存优化】Bitmap 图像标准缩小 ( 设置 Options 参数 | inJustDecodeBounds | inSampleSize | 东西类结束 ) 一、解码图片参数 inJustDecodeBounds 章节内容 , 有图片解码的具体步骤 ;




1 . 图片标准数据解码 :



① 创立解码选项 : 创立 BitmapFactory.Options 政策 ;


② 设置解码标准数据 : 设置 BitmapFactory.Options 政策的 inJustDecodeBounds 为 true , 解码图像时 , 不解码图像数据 , 只获取图像的标准数据 ;


③ 解码图像标准数据 : 调用 BitmapFactory.decodeStream 办法 , 解码图片 , 图片相关的标准数据保存到了 mOptions 选项中 ;


④ 获取图片标准 : mOptions.outWidth 是解码出的图像宽度 , mOptions.outHeight 是解码出的图像高度 ;




2 . 代码示例 :


/**  * Bitmap 解码选项  */ private BitmapFactory.Options mOptions; /**  * 图片宽度  */ private int mImageWidth; /**  * 图片高度  */ private int mImageHeight; // ... // 解码选项 mOptions = new BitmapFactory.Options(); // 读取图片的标准数据 mOptions.inJustDecodeBounds = true; // 解码图片 , 图片相关的标准数据保存到了 mOptions 选项中 BitmapFactory.decodeStream(inputStream, null, mOptions); // 获取图片宽高 mImageWidth = mOptions.outWidth; mImageHeight = mOptions.outHeight;




二、核算解码区域


1 . 闪现区域核算准则 : 这是一张长图 , 宽度彻底闪现 , 高度闪现部分 ; 根据组件的宽高核算图像闪现的区域 , 组件的宽高已知 , 宽高比例供认 ; 该宽高比例下 , 图片闪现的区域也有必要是该比例 ;




2 . 图像宽高与组件宽高比例 : 加载的图像高度宽度 , 与组件的高度宽度比例一同 ;



m V i e w W i d t h m V i e w H e i g h t = 加 载 的 图 像 宽 度 加 载 的 图 像 高 度 \dfrac{mViewWidth }{mViewHeight} = \dfrac{加载的图像宽度}{加载的图像高度}

mViewHeight

mViewWidth


=

加载的图像高度

加载的图像宽度




m V i e w W i d t h 加 载 的 图 像 宽 度 = m V i e w H e i g h t 加 载 的 图 像 高 度 \dfrac{mViewWidth }{加载的图像宽度} = \dfrac{mViewHeight }{加载的图像高度}

加载的图像宽度

mViewWidth


=

加载的图像高度

mViewHeight





3 . 缩放因子 : 由于宽度有必要填充慢组件宽度 , 这儿需求缩放图片 , 高分辨率手机需求缩小图片 , 低分辨率手机需求扩展图片 ;


缩 放 因 子 = m V i e w W i d t h 加 载 的 图 像 宽 度 缩放因子 = \dfrac{mViewWidth}{加载的图像宽度 }

缩放因子=

加载的图像宽度

mViewWidth





4 . 核算区域高度 : 图像截取的宽度已知 , 组件的宽高已知 , 核算图像截取的高度 :


m V i e w W i d t h 加 载 的 图 像 宽 度 = m V i e w H e i g h t 加 载 的 图 像 高 度 = m V i e w H e i g h t × 加 载 的 图 像 宽 度 m V i e w W i d t h = m V i e w H e i g h t 缩 放 因 子

mViewWidth加载的图像宽度===mViewHeight加载的图像高度mViewHeight×加载的图像宽度mViewWidthmViewHeight缩放因子

mViewWidth加载的图像宽度=mViewHeight加载的图像高度=mViewHeight×加载的图像宽度mViewWidth=mViewHeight缩放因子

加载的图像宽度

mViewWidth




=

=

=



加载的图像高度

mViewHeight


mViewWidth

mViewHeight×加载的图像宽度


缩放因子

mViewHeight






5 . 代码示例 : 在 onMeasure 办法中 , 获取最新测量出来的组件宽高 , 根据以上公式 , 核算出要解码图像的宽高 ;


   @Override


protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  super.onMeasure(widthMeasureSpec, heightMeasureSpec);  // 获取测量的自定义 View 组件宽高  mViewWidth = getMeasuredWidth();  mViewHeight = getMeasuredHeight();  // 根据组件的宽高 , 供认要加载的图像的宽高  if(mBitmapRegionDecoder != null){  mRect.left = 0;  mRect.top = 0;  // 制造的宽度便是图像的宽度  mRect.right = mImageWidth;  // 根据图像宽度 和 组件宽度 , 核算出缩放比例  // 组件宽度 / 图像宽度 = 缩放因子  mScale = (float)mViewWidth / (float)mImageWidth;  /*  加载的图像高度宽度 , 与组件的高度宽度比例一同  mViewWidth / 加载的图像宽度 = mViewHeight / 加载的图像高度  此处加载的图像宽度便是实践的宽度  加载的图像高度 = mViewHeight / ( mViewWidth / 加载的图像宽度 )  mViewWidth / 加载的图像宽度 便是缩放因子  加载的图像高度 = mViewHeight / 缩放因子  */  // 根据缩放因子核算解码高度  mRect.bottom = (int) (mViewHeight / mScale);  }  }







三、设置解码参数 内存复用 像素格式


设置图像解码参数 :



① 关闭标准解码 : 之前解码图像标准 , 将 BitmapFactory.Options 的 inJustDecodeBounds 特征设置为了 true , 现在要初步解码图像数据了 , 需求关闭该选项 , 设置为 false ;


② 设置像素格式 : 假定不需求闪现透明度 , 就设置 BitmapFactory.Options 的 inPreferredConfig 像素格式为 Bitmap.Config.RGB_565 , 该像素格式每个像素占 2 字节内存 ;


③ 设置可变 : 这是内存复用收效的条件 , 设置 inMutable 为 true ;


④ 设置复用内存的 Bitmap 政策 : 每次解码操作前都要设置一次 , 解码时会复用该 Bitmap 中的内存 ;




2 . 代码示例 :

/**  * Bitmap 解码选项  */ private BitmapFactory.Options mOptions; // ... // 设置 Bitmap 内存复用 mOptions.inMutable = true;  // 设置可变 // 内存复用 mOptions.inBitmap = mBitmap; mOptions.inPreferredConfig = Bitmap.Config.RGB_565; // 设置像素格式 RGB 565 mOptions.inJustDecodeBounds = false; // 读取结束之后, 就需求解析实践的 Bitmap 图像数据了






四、图像制造


1 . 图像制造 :



① 设置图像区域解码器 : 在为自定义组件设置图片时 , 设置区域解码器 , 由于要设置区域解码的数据源 , 因而有必要在用户设置图片时 , 才能够创立区域解码器 ;


② 设置内存复用 : 每次解码时 , 都要设置一下内存复用的 Bitmap 政策 ; mOptions.inBitmap = mBitmap;


③ 解码图片 : 调用区域解码器的 mBitmapRegionDecoder.decodeRegion 办法 , 解码图片的特定区域 ;


④ 设置图片缩放 : 运用 Matrix 进行图像缩放 ; 图像与自定义组件的标准不同 , 因而需求将解码区域彻底填充到自定义组件中闪现 ;


⑤ 图像制造 : 调用 canvas.drawBitmap 制造图像 , 假定需求缩放 , 传入 Matrix 参数 ;




2 . 代码示例 :



/**  * 图像区域解码器  */  private BitmapRegionDecoder mBitmapRegionDecoder;  // ...  /**  * 设置闪现的图片  * @param inputStream  */  public void setImage(InputStream inputStream){  // ...  try {  // Bitmap 区域解码器  mBitmapRegionDecoder = BitmapRegionDecoder.newInstance(inputStream, false);  } catch (IOException e) {  e.printStackTrace();  }  // 设置图片结束后 , 改写自定义组件  requestLayout();  }  // ...  @Override  protected void onDraw(Canvas canvas) {  super.onDraw(canvas);  if(mBitmapRegionDecoder == null) return;  // 内存复用  mOptions.inBitmap = mBitmap;  // 解码图片  mBitmap = mBitmapRegionDecoder.decodeRegion(mRect, mOptions);  // 设置制造的图像缩放 , x 轴和 y 轴都在 Bitmap 大小的区域基础上 , 缩放 mScale 倍  Matrix matrix = new Matrix();  matrix.setScale(mScale, mScale);  canvas.drawBitmap(mBitmap, matrix, null);  }




五、施行效果


竖屏效果 :

image.png



横屏效果 :



image.png





六、源码及资源下载


源码及资源下载地址 :


① GitHub 工程地址 : Long_Graph_Loading


② LongImageView.java 主界面代码地址 : LongImageView.java , 这是上述示自定义组件代码 ;


12年经验 · 提供上云保障

服务热线:132-6161-6125(手机) 400-150-1900(全国市话)

站内导航: 天翼云服务器价格| 天翼云购买流程| 天翼云代理| 北京天翼云代理商| 杭州天翼云代理| 深圳天翼云代理商| 钉钉代理商| 阿里云代理| 公司官网

我公司收款账号| 天翼云备案系统

CopyRight © 2019 天翼云代理商. All Rights Reserved 京ICP备2024071855号-7 管理-北京优胜智连科技有限公司