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

Android BitmapRegionDecoder 详解

Android BitmapRegionDecoder 详解

BitmapRegionDecoder 是 Android 提供的一个用于局部解码大图的类,适用于加载超大型图片(如高清地图、长图、全景照片等),避免一次性加载整个图片导致内存溢出(OOM)。


1. 基本使用

(1) 初始化 BitmapRegionDecoder

可以从 InputStreamFileDescriptorByteArray 创建:

val inputStream = context.assets.open("huge_image.jpg")
val decoder = BitmapRegionDecoder.newInstance(inputStream, false)

⚠️ 注意false 表示不共享输入流,解码完成后需手动关闭 inputStream

(2) 解码指定区域

val rect = Rect(left, top, right, bottom) // 要解码的区域坐标
val options = BitmapFactory.Options().apply {inPreferredConfig = Bitmap.Config.RGB_565 // 减少内存占用inSampleSize = 2 // 可选:进一步缩放
}
val regionBitmap = decoder.decodeRegion(rect, options)
imageView.setImageBitmap(regionBitmap)
  • Rect:指定要解码的矩形区域(单位:像素)。
  • inSampleSize:缩放系数(2 表示宽高各缩小一半)。
  • inPreferredConfig
    • Bitmap.Config.ARGB_8888(默认,每个像素占 4 字节)
    • Bitmap.Config.RGB_565(推荐,每个像素占 2 字节,适合不透明图片)

(3) 释放资源

decoder.recycle() // 不再使用时释放内存
regionBitmap?.recycle() // 手动回收 Bitmap

2. 典型应用场景

(1) 超大图片分块加载(如地图)

// 根据 ImageView 当前显示区域动态加载
fun loadVisibleRegion(imageView: ImageView, decoder: BitmapRegionDecoder) {val visibleRect = getVisibleRect(imageView) // 计算可见区域val options = BitmapFactory.Options().apply {inPreferredConfig = Bitmap.Config.RGB_565}val regionBitmap = decoder.decodeRegion(visibleRect, options)imageView.setImageBitmap(regionBitmap)
}

(2) 高清长图浏览(类似微博长图)

结合 ImageViewOnTouchListenerRecyclerView,动态解码当前屏幕区域:

imageView.setOnTouchListener { _, event ->when (event.action) {MotionEvent.ACTION_MOVE -> {val visibleRect = calculateVisibleRect(event) // 根据手势计算新区域val newRegion = decoder.decodeRegion(visibleRect, options)imageView.setImageBitmap(newRegion)}}true
}

3. 优化技巧

(1) 复用 Bitmap 对象

避免频繁创建/回收 Bitmap,使用 BitmapPool(如 Glide 的实现):

val options = BitmapFactory.Options().apply {inBitmap = reusableBitmap // 复用已有 Bitmap 内存
}

(2) 内存缓存

使用 LruCache 缓存已解码的局部区域:

private val regionCache = LruCache<String, Bitmap>(maxSize)fun getCachedRegion(key: String): Bitmap? {return regionCache.get(key)
}fun cacheRegion(key: String, bitmap: Bitmap) {regionCache.put(key, bitmap)
}

(3) 异步加载

配合 CoroutineRxJava 防止 UI 卡顿:

viewModelScope.launch(Dispatchers.IO) {val regionBitmap = decoder.decodeRegion(rect, options)withContext(Dispatchers.Main) {imageView.setImageBitmap(regionBitmap)}
}

4. 对比其他方案

方案适用场景优点缺点
BitmapRegionDecoder超大图局部加载精准控制内存占用需手动计算区域坐标
Glide/Picasso常规图片加载自动缓存、生命周期管理不支持局部解码
SubsamplingScaleImageView开源库(支持手势缩放)功能完善增加 APK 体积

5. 完整示例代码

class BigImageActivity : AppCompatActivity() {private lateinit var decoder: BitmapRegionDecoderprivate var currentBitmap: Bitmap? = nulloverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_big_image)// 初始化解码器val inputStream = assets.open("huge_image.jpg")decoder = BitmapRegionDecoder.newInstance(inputStream, false)inputStream.close()// 首次加载中心区域loadInitialRegion()imageView.setOnTouchListener(gestureListener)}private fun loadInitialRegion() {val centerX = decoder.width / 2val centerY = decoder.height / 2val rect = Rect(centerX - 500, centerY - 500,centerX + 500, centerY + 500)currentBitmap?.recycle()currentBitmap = decoder.decodeRegion(rect, BitmapFactory.Options().apply {inPreferredConfig = Bitmap.Config.RGB_565})imageView.setImageBitmap(currentBitmap)}private val gestureListener = object : View.OnTouchListener {override fun onTouch(v: View, event: MotionEvent): Boolean {// 实现手势滑动逻辑(略)return true}}override fun onDestroy() {decoder.recycle()currentBitmap?.recycle()super.onDestroy()}
}

总结

  • 使用场景BitmapRegionDecoder 适合加载无法缩放的全尺寸大图(如高清地图、医学影像)。
  • 关键优化:通过 Rect 控制解码范围 + inSampleSize 缩放 + RGB_565 配置。
  • 进阶方案:结合手势库(如 PhotoView)实现交互式浏览。
http://www.lryc.cn/news/579799.html

相关文章:

  • Java启动脚本
  • vue create 和npm init 创建项目对比
  • error MSB8041: 此项目需要 MFC 库。从 Visual Studio 安装程序(单个组件选项卡)为正在使用的任何工具集和体系结构安装它们。
  • React 渲染深度解密:从 JSX 到 DOM 的初次与重渲染全流程
  • 最快实现的前端灰度方案
  • 因果语言模型、自回归语言模型、仅解码器语言模型都是同一类模型
  • 同步(Synchronization)和互斥(Mutual Exclusion)关系
  • 【机器人】复现 DOV-SG 机器人导航 | 动态开放词汇 | 3D 场景图
  • (超详细)数据库项目初体验:使用C语言连接数据库完成短地址服务(本地运行版)
  • 敏捷开发在国际化团队管理中的落地
  • 二维码驱动的独立站视频集成方案
  • 碰一碰发视频源码搭建与定制化开发:支持OEM
  • 译码器Multisim电路仿真汇总——硬件工程师笔记
  • TensorFlow 安装使用教程
  • MySQL数据库----DML语句
  • 【2.4 漫画SpringBoot实战】
  • 【模糊集合】示例
  • vue-37(模拟依赖项进行隔离测试)
  • Unity Android与iOS自动重启
  • uniapp打包微信小程序主包过大问题_uniapp 微信小程序时主包太大和vendor.js过大
  • 《导引系统原理》-西北工业大学-周军-“2️⃣导引头的角度稳定系统”
  • 暑期前端训练day3
  • RNN案例人名分类器(完整步骤)
  • Linux常见指令以及权限理解
  • 网安系列【1】:黑客思维、技术与案例解析
  • 实现如何利用 Kafka 延时删除 用户邮箱的验证码(如何发送邮箱+源码) - 第一期
  • Web攻防-文件上传黑白名单MIMEJS前端执行权限编码解析OSS存储分域名应用场景
  • 二叉树题解——二叉树的层序遍历【LeetCode】队列实现
  • 热血三国建筑攻略表格
  • SciPy 安装使用教程