glide ModelLoader的Key错误使用 可能造成的内存泄漏

glide ModelLoader的Key错误使用 可能造成的内存泄漏

业务场景

之前项目性能优化,在自定义的AppGlideModule中的registerComponents方法append了自定义ModelLoaderFactory,然后有很多个File对象出现了内存泄漏,后面定位到以下场景,代码如下:

kotlin 复制代码
import com.bumptech.glide.load.Options
import com.bumptech.glide.load.model.ModelLoader
import com.bumptech.glide.signature.ObjectKey
import com.fawvw.hmi.media.mediacommon.model.mediadata.AudioMediaData
import java.io.InputStream

class AudioModuleLoader : ModelLoader<AudioMediaData, InputStream> {
    override fun buildLoadData(
        model: AudioMediaData,
        width: Int,
        height: Int,
        options: Options
    ): ModelLoader.LoadData<InputStream> {
        val path = if (model.albumArt != null) {
            "${model.artist}${model.duration}${model.album}"
        } else {
            model.uri
        }
        return ModelLoader.LoadData(ObjectKey(path), AudioCoverFetcher(model))
    }
	//这里使用了AudioCoverSignature
	 return ModelLoader.LoadData(AudioCoverSignature(path), AudioCoverFetcher(model))
    }

    override fun handles(model: AudioMediaData) = true
}
kotlin 复制代码
import com.bumptech.glide.load.Key
import java.io.File
import java.security.MessageDigest

class AudioCoverSignature(private val path:String): Key {
    private val stringBuilder by lazy {
        StringBuilder()
    }
	//这个file内存泄漏了
    private val file by lazy {
        File(path)
    }
    override fun updateDiskCacheKey(messageDigest: MessageDigest) {
        stringBuilder.append(file.lastModified()).append(file.absolutePath)
        val bs = stringBuilder.toString().encodeToByteArray()
        messageDigest.update(bs, 0, bs.size)
    }
}

泄露的原因是因为,AudioCoverSignature类只重写了updateDiskCacheKey方法,没有重写equals方法,而ModelLoader.LoadData方法的入参key是用来找缓存的,每一个都找不到,导致内存泄漏

kotlin 复制代码
  /**
   * Contains a set of {@link com.bumptech.glide.load.Key Keys} identifying the source of the load,
   * alternate cache keys pointing to equivalent data, and a {@link
   * com.bumptech.glide.load.data.DataFetcher} that can be used to fetch data not found in cache.
   *
   * @param <Data> The type of data that well be loaded.
   */
  class LoadData<Data> {
    public final Key sourceKey;
    public final List<Key> alternateKeys;
    public final DataFetcher<Data> fetcher;
	...

解决方法

将自定义AudioCoverSignature删除,使用ObjectKey解决此问题

kotlin 复制代码
import com.bumptech.glide.load.Options
import com.bumptech.glide.load.model.ModelLoader
import com.bumptech.glide.signature.ObjectKey
import com.fawvw.hmi.media.mediacommon.model.mediadata.AudioMediaData
import java.io.InputStream

class AudioModuleLoader : ModelLoader<AudioMediaData, InputStream> {
    override fun buildLoadData(
        model: AudioMediaData,
        width: Int,
        height: Int,
        options: Options
    ): ModelLoader.LoadData<InputStream> {
        val path = if (model.albumArt != null) {
            "${model.artist}${model.duration}${model.album}"
        } else {
            model.uri
        }
		//这里使用ObjectKey
        return ModelLoader.LoadData(ObjectKey(path), AudioCoverFetcher(model))
    }

    override fun handles(model: AudioMediaData) = true
}

问题解决了

相关推荐
2501_915918414 分钟前
iOS性能数据监控:从概念到工具实践,让应用运行更流畅
android·macos·ios·小程序·uni-app·cocoa·iphone
帅次14 分钟前
测试分层:JVM 单测、ViewModel 测试与 Compose UI Test
android·jvm·ui·kotlin·compose·modifier
背包客(wyq)16 分钟前
开源中文语音模型Android端部署测试
android·开源
yewq-cn16 分钟前
Android Compatibility
android
马占柱24 分钟前
MediaSession相关介绍
android·多媒体·mediasession·车载多媒体开发
张风捷特烈33 分钟前
状态管理大乱斗#08 | MobX 源码评析 - 透明魔法
android·前端·flutter
i220818 Faiz Ul33 分钟前
个人健康系统|健康管理|基于java+Android+微信小程序的个人健康系统设计与实现(源码+数据库+文档)
android·java·vue.js·spring boot·微信小程序·毕设·个人健康系统
赏金术士9 小时前
Kotlin 习题集 · 高级篇
android·开发语言·kotlin
问心无愧051311 小时前
ctf show web 入门42
android·前端·android studio
没什么本事12 小时前
关于C# panel 添加lable问题 -- 明确X和Y 位置错误
android·java·c#