Android 大图显示策略优化显示(二)

Glide有2种获取图片宽高的方式:

通过添加 RequestListener 并设置 override(Target.SIZE_ORIGINAL),可以在图片解码前获取其原始尺寸信息。

复制代码
Glide.with(context)
    .asBitmap()
    .load(R.drawable.your_image) // 或 Uri、URL、文件路径
    .override(Target.SIZE_ORIGINAL) // 请求原始尺寸
    .addListener(object : RequestListener<Bitmap> {
        override fun onResourceReady(
            resource: Bitmap?,
            model: Any?,
            target: Target<Bitmap>?,
            dataSource: DataSource?,
            isFirstResource: Boolean
        ): Boolean {
            resource?.let {
                val width = it.width
                val height = it.height
                // 在此处处理获取到的宽高
                Log.d("GlideSize", "Width: $width, Height: $height")
            }
            return true // 返回 true 阻止图片传递给 ImageView
        }

        override fun onLoadFailed(
            e: GlideException?,
            model: Any?,
            target: Target<Bitmap>?,
            isFirstResource: Boolean
        ): Boolean {
            return false
        }
    })
    .preload() // 仅预加载,不显示

第二种方式:

通过 SimpleTarget 的回调获取尺寸后立即释放资源,避免内存占用

复制代码
Glide.with(context.applicationContext) // 使用 ApplicationContext 避免生命周期问题
    .asBitmap()
    .load(R.drawable.your_image)
    .into(object : SimpleTarget<Bitmap>() {
        override fun onResourceReady(
            resource: Bitmap,
            transition: Transition<in Bitmap>?
        ) {
            val width = resource.width
            val height = resource.height
            // 处理获取到的宽高
            resource.recycle() // 获取尺寸后立即回收 Bitmap
        }
    })
  1. 性能优化 ‌:方法一通过 override(Target.SIZE_ORIGINAL)preload() 组合,能最大程度减少内存使用,仅解码图片头部信息获取尺寸1。
  2. 生命周期管理 ‌:如果仅需获取尺寸而不涉及 UI 更新,建议使用 ApplicationContext 以避免内存泄漏2。
  3. 网络图片‌:对于网络图片,Glide 会先下载图片头部信息来获取尺寸,不会下载完整图片数据1。
  4. 回调处理‌:两种方法均为异步操作,获取到的宽高需在回调中处理。

根据上篇文章的思想继续优化:

复制代码
private fun loadImageToView(
        context: Context,
        maxWidth: Int,
        maxHeight: Int,
        resId: Int,
        igvTarget: ImageView
    ) {
        Glide.with(context)
            .asBitmap()
            .load(resId) // 或 Uri、URL、文件路径
            .override(Target.SIZE_ORIGINAL) // 请求原始尺寸
            .addListener(object : RequestListener<Bitmap> {
                override fun onResourceReady(
                    resource: Bitmap,
                    model: Any,
                    target: com.bumptech.glide.request.target.Target<Bitmap>,
                    dataSource: DataSource,
                    isFirstResource: Boolean
                ): Boolean {
                    resource?.let {
                        val width = it.width
                        val height = it.height
                        // 在此处处理获取到的宽高
                        Log.d("GlideSize", "Width: $width, Height: $height")
                        val scaleResult =
                            ImageUtils.calculateScaleParameters(
                                maxWidth,
                                maxHeight,
                                width,
                                height
                            )
                        runOnUiThread {
                            igvTarget.layoutParams.height = scaleResult.newHeight
                            igvTarget.layoutParams.width = scaleResult.newWidth
                            Glide.with(context).load(resId).into(igvTarget)
                        }
                    }
                    return true // 返回 true 阻止图片传递给 ImageView
                }

                override fun onLoadFailed(
                    e: GlideException?,
                    model: Any,
                    target: com.bumptech.glide.request.target.Target<Bitmap>,
                    isFirstResource: Boolean
                ): Boolean {
                    return false
                }
            })
            .preload() // 仅预加载,不显示
    }

    private fun loadImageToView(
        context: Context,
        maxWidth: Int,
        maxHeight: Int,
        path: String,
        igvTarget: ImageView
    ) {
        Glide.with(context)
            .asBitmap()
            .load(path) // 或 Uri、URL、文件路径
            .override(Target.SIZE_ORIGINAL) // 请求原始尺寸
            .addListener(object : RequestListener<Bitmap> {
                override fun onResourceReady(
                    resource: Bitmap,
                    model: Any,
                    target: com.bumptech.glide.request.target.Target<Bitmap>,
                    dataSource: DataSource,
                    isFirstResource: Boolean
                ): Boolean {
                    resource?.let {
                        val width = it.width
                        val height = it.height
                        // 在此处处理获取到的宽高
                        Log.d("GlideSize", "Width: $width, Height: $height")
                        val scaleResult =
                            ImageUtils.calculateScaleParameters(
                                maxWidth,
                                maxHeight,
                                width,
                                height
                            )
                        runOnUiThread {
                            igvTarget.layoutParams.height = scaleResult.newHeight
                            igvTarget.layoutParams.width = scaleResult.newWidth
                            Glide.with(context).load(path).into(igvTarget)
                        }
                    }
                    return true // 返回 true 阻止图片传递给 ImageView
                }

                override fun onLoadFailed(
                    e: GlideException?,
                    model: Any,
                    target: com.bumptech.glide.request.target.Target<Bitmap>,
                    isFirstResource: Boolean
                ): Boolean {
                    return false
                }
            })
            .preload() // 仅预加载,不显示
    }
相关推荐
AI人工智能+电脑小能手11 小时前
【大白话说Java面试题】【Java基础篇】第25题:JDK1.8的新特性有哪些
java·开发语言·后端·面试
likerhood12 小时前
SLF4J: Failed to load class “StaticLoggerBinder“ 解决
java·log4j·maven
开发小程序的之朴12 小时前
基于Go语言的企业级CMS系统架构设计与性能分析——以AnQiCMS为例
开发语言·golang·系统架构
早日退休!!!12 小时前
大模型推理瓶颈七层分析模型
java·服务器·数据库
叶小鸡12 小时前
Java 篇-项目实战-天机学堂(从0到1)-day9
java·开发语言
兔子零102412 小时前
Ofox AI值得用吗?
前端·javascript·后端
小短腿的代码世界12 小时前
Qt券商接口封装深度解析:统一API设计与多源适配
开发语言·qt·单元测试
wearegogog12312 小时前
基于Q-learning的栅格地图路径规划MATLAB仿真程序
开发语言·算法·matlab
@#¥&~是乱码鱼啦12 小时前
Spring分层架构:Controller、Service、Mapper数据链路,IOC的真实工作意义
java·spring·架构
jinanwuhuaguo12 小时前
OpenClaw联邦之心——从孤岛记忆到硅基集体潜意识的拓扑学革命(第二十三篇)
android·人工智能·kotlin·拓扑学·openclaw