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
}

问题解决了

相关推荐
烬奇小云2 小时前
认识一下Unicorn
android·python·安全·系统安全
顾北川_野14 小时前
Android 进入浏览器下载应用,下载的是bin文件无法安装,应为apk文件
android
CYRUS STUDIO14 小时前
Android 下内联汇编,Android Studio 汇编开发
android·汇编·arm开发·android studio·arm
右手吉他14 小时前
Android ANR分析总结
android
PenguinLetsGo16 小时前
关于 Android15 GKI2407R40 导致梆梆加固软件崩溃
android·linux
杨武博19 小时前
音频格式转换
android·音视频
音视频牛哥21 小时前
Android音视频直播低延迟探究之:WLAN低延迟模式
android·音视频·实时音视频·大牛直播sdk·rtsp播放器·rtmp播放器·android rtmp
ChangYan.21 小时前
CondaError: Run ‘conda init‘ before ‘conda activate‘解决办法
android·conda
二流小码农21 小时前
鸿蒙开发:ForEach中为什么键值生成函数很重要
android·ios·harmonyos
夏非夏1 天前
Android 生成并加载PDF文件
android