Android Coil ImageLoader MemoryCache设置Key与复用内存缓存,Kotlin

Android Coil ImageLoader MemoryCache设置Key与复用内存缓存,Kotlin

Kotlin 复制代码
import android.content.ContentUris
import android.content.Context
import android.net.Uri
import android.os.Bundle
import android.provider.MediaStore
import android.util.Log
import androidx.appcompat.app.AppCompatActivity
import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.GridLayoutManager
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch

class MainActivity : AppCompatActivity() {
    companion object {
        const val TAG = "fly"
        const val SPAN_COUNT = 8
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.rv_layout)

        val rv = findViewById<RecyclerView>(R.id.rv)

        val layoutManager = GridLayoutManager(this, SPAN_COUNT)
        layoutManager.orientation = LinearLayoutManager.VERTICAL
        val adapter = ImageAdapter(this, 0)
        rv.adapter = adapter
        rv.layoutManager = layoutManager

        //rv.setHasFixedSize(true)
        //rv.setItemViewCacheSize(SPAN_COUNT * 20)
        //rv.recycledViewPool.setMaxRecycledViews(0, SPAN_COUNT * 20)

        val ctx = this
        lifecycleScope.launch(Dispatchers.IO) {
            val lists = readAllImage(ctx)
            Log.d(TAG, "readAllImage size=${lists.size}")

            lifecycleScope.launch(Dispatchers.Main) {
                adapter.dataChanged(lists)
            }
        }
    }

    class MyData(var path: String, var uri: Uri)

    private fun readAllImage(ctx: Context): ArrayList<MyData> {
        val photos = ArrayList<MyData>()

        //读取所有图
        val cursor = ctx.contentResolver.query(
            MediaStore.Images.Media.EXTERNAL_CONTENT_URI, null, null, null, null
        )

        while (cursor!!.moveToNext()) {
            //路径
            val path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA))

            val id = cursor.getColumnIndex(MediaStore.Images.ImageColumns._ID)
            val imageUri: Uri = ContentUris.withAppendedId(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, cursor.getLong(id))

            //名称
            //val name = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DISPLAY_NAME))

            //大小
            //val size = cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.SIZE))

            photos.add(MyData(path, imageUri))
        }
        cursor.close()

        return photos
    }
}
Kotlin 复制代码
import android.content.Context
import android.graphics.Bitmap
import android.util.Log
import android.view.View
import android.view.ViewGroup
import androidx.appcompat.widget.AppCompatImageView
import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.RecyclerView
import coil.ImageLoader
import coil.memory.MemoryCache
import coil.request.CachePolicy
import coil.request.ImageRequest
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch


class ImageAdapter : RecyclerView.Adapter<ImageHolder> {
    private var mCtx: Context? = null
    private var mImageLoader: ImageLoader? = null
    private var mViewSize = 0

    constructor(ctx: Context, type: Int) : super() {
        mCtx = ctx

        mImageLoader = ImageLoader.Builder(mCtx!!)
            .memoryCachePolicy(CachePolicy.ENABLED)
            .availableMemoryPercentage(0.9999)
            .bitmapPoolingEnabled(true)
            .bitmapConfig(Bitmap.Config.RGB_565)
            .diskCachePolicy(CachePolicy.ENABLED)
            .build()

        Log.d(MainActivity.TAG, "memoryCache.maxSize=${mImageLoader?.memoryCache?.maxSize}")

        mViewSize = mCtx!!.resources.displayMetrics.widthPixels / MainActivity.SPAN_COUNT
    }

    private var mItems = ArrayList<MainActivity.MyData>()

    fun dataChanged(items: ArrayList<MainActivity.MyData>) {
        this.mItems = items
        notifyDataSetChanged()
    }

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ImageHolder {
        val view = MyIV(mCtx!!, mViewSize)

        return ImageHolder(view)
    }

    override fun getItemCount(): Int {
        return mItems.size
    }

    override fun onBindViewHolder(holder: ImageHolder, position: Int) {
        bind(mItems[position], holder.image)
    }

    private fun bind(data: MainActivity.MyData, image: MyIV) {
        val key = MemoryCache.Key.invoke(data.path)

        val bmp = mImageLoader?.memoryCache?.get(key)

        if (bmp != null && bmp.byteCount > 0) {
            Log.d(
                MainActivity.TAG,
                "memory cache bmp=${bmp.byteCount} ${mImageLoader?.memoryCache?.size}/${mImageLoader?.memoryCache?.maxSize}"
            )
            image.setImageBitmap(bmp)
        } else {
            val t = System.currentTimeMillis()

            val request = ImageRequest.Builder(mCtx!!)
                .data(data.uri)
                .size(mViewSize)
                .target(image)
                .memoryCacheKey(key)
                .build()

            (mCtx as? MainActivity)?.lifecycleScope?.launch(Dispatchers.IO) {
                mImageLoader?.execute(request)
            }

            Log.d(MainActivity.TAG, "no memory cache time=${System.currentTimeMillis() - t}")
        }
    }
}

class ImageHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
    var image = itemView as MyIV
}

class MyIV : AppCompatImageView {
    private var mSize = 0

    constructor(ctx: Context, size: Int) : super(ctx) {
        mSize = size
        scaleType = ScaleType.CENTER_CROP
    }

    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec)

        setMeasuredDimension(mSize, mSize)
    }
}

1、已知问题,发现Coil在宫格多时候,如果直接:

Kotlin 复制代码
mImageLoader?.enqueue(request)

也有一定小的耗时。所以干脆用协程包装起来。

2、虽然通过设置内存系数

Kotlin 复制代码
availableMemoryPercentage

扩大了内存,但跑起来发现设置后内存还是比较小(约300mb),这是不够的,需要通过其他配置方式扩大内存空间。

3、app跑起来后,没有在当前app的硬盘缓存空间发现图片解码后的磁盘文件缓存痕迹。这需要再配置。

Android图片加载框架Coil,Kotlin-CSDN博客文章浏览阅读649次。Coil是专门针对Android平台上的Kotlin语言特性设计,这不像Glide,Glide的核心框架语言是Java。Coil实现看更细颗粒度的内存、磁盘缓存的客制化设置。https://blog.csdn.net/zhangphil/article/details/145558324

相关推荐
SRC_BLUE_1740 分钟前
NSSCTF - Web | 【第五空间 2021】pklovecloud
android·前端
奥陌陌1 小时前
kotlin className.() 类名点花括号 T.() 这种是什么意思?
kotlin
tq10862 小时前
学习Hilt注解
android
2501_915921433 小时前
iOS 应用代上架流程,多工具组合与使用 开心上架 跨平台自动化上传指南
android·ios·小程序·uni-app·自动化·cocoa·iphone
日日行不惧千万里3 小时前
2025最新仿默往 IM 即时通讯系统源码(PC + Web + iOS + Android)完整版发布!
android·ios
歪歪1003 小时前
React Native开发Android&IOS流程完整指南
android·开发语言·前端·react native·ios·前端框架
雪芽蓝域zzs3 小时前
uniapp 修改android包名
android·uni-app
用户2018792831673 小时前
厨房里的协程大冒险:launch与async的烹饪之旅
android
用户2018792831673 小时前
浅析协程与挂起函数实现原理
android
木易士心4 小时前
Android Handler 机制原理详解
android·app