最近任务图标,缩略图的加载和缓存

简述

Screenshot_20181026-094023.png

    最近任务卡片上的图片资源主要有应用图标和缩略图。应用图标一般是固定的,加载后就放在缓存里,不需要每次从系统里加载。缩略图则有可能会经常变化,所以经常需要实时加载,加载的时候一般使用异步加载。
    图标和缩略图从加载到显示在卡片上的流程可以分为三部分。
        1. 从系统预加载资源
        2. 界面绘图
        3. 线程异步加载资源,加载后更新到界面
        4. Android缓存策略——LruCache

1. 从系统预加载资源

最近任务模块启动后,会在两个回调函数中从系统预加载资源。分别是
        RecentsImpl.onBootComplete
        android.app.TaskStackListener mTaskStackListener.onTaskStackChanged

onBootComplete函数是在系统启动完成后被调用的,onTaskStackChanged是在ActivityManager发现系统的TaskStack发生变化以后调用的。
这两个函数调用RecentsTaskLoader.loadTasks函数以同步的方式从系统加载资源。
loadTasks最终会调用两个函数分别加载图标和缩略图。
加载图标的函数是:RecentsTaskLoader.getAndUpdateActivityIcon
加载缩略图的函数是:RecentsTaskLoader.getAndUpdateThumbnail

这两个函数首先从缓存中查找图标,如果缓存中有,并且没有失效,则使用缓存中的图标,否则从系统中获取。
图标缓存类是TaskKeyLruCache,缩略图缓存类是TaskKeyStrongCache,这两个类都是TaskKeyCache的子类。TaskKeyCache通过比较Task.TaskKey.lastActiveTime来确定缓存是否失效

    final V getAndInvalidateIfModified(Task.TaskKey key) {
        Task.TaskKey lastKey = mKeys.get(key.id);
        if (lastKey != null) {
            if ((lastKey.stackId != key.stackId) ||
                    (lastKey.lastActiveTime != key.lastActiveTime)) {
                // The task has updated (been made active since the last time it was put into the
                // LRU cache) or the stack id for the task has changed, invalidate that cache item
                remove(key);
                return null;
            }
        }
        // Either the task does not exist in the cache, or the last active time is the same as
        // the key specified, so return what is in the cache
        return getCacheEntry(key.id);
    }

系统刚启动完成的时候,缓存里是没有资源的,所以这个时候一般都是从系统加载资源。

2. 界面绘图

第一步从系统加载资源后,界面绘图就要开始使用这些资源了。
使用资源是通过异步方式进行的。

TaskStackView.onMeasure  
    TaskStackView.bindVisibleTaskViews
        ViewPool.pickUpViewFromPool
            TaskStackView.createView
            TaskStackView.onPickUpViewFromPool
                TaskStackView.updateTaskViewsList //更新mTaskViews
                TaskStackView.bindTaskView // 绑定taskView和应用图标以及缩略图
                    RecentsTaskLoader.loadTaskData // 把Task加入到mLoadQueue中,RecentsTaskLoader以异步方式加载应用图标和缩略图

在TaskStackView测量的时候,会根据task列表创建TaskView,为每个TaskView提交一个加载任务到RecentsTaskLoader。RecentsTaskLoader以异步的方式加载图标和缩略图。

3. 线程异步加载资源,加载后更新到界面

RecentsView被绘制之前,会启动一个后台线程mLoadThread,这个线程循环读取mLoadQueue,为每一个task加载资源。

mRecentsDrawnEventListener                                      //RecentsView视图树绘制前的回调接口
    Recents.getTaskLoader().startLoader(RecentsActivity.this);  //启动RecentsTaskLoader的加载线程
      BackgroundTaskLoader.start
        mLoadThreadHandler.post(this)  //投递到HandlerThread线程, mLoadThreadHandler是HandlerThread的handler
          RecentsTaskLoader.processLoadQueueItem
            取cachedIcon
              mIconCache.get(t.key)
              ssp.getBadgedTaskDescriptionIcon
              ssp.getActivityInfo(t.key.getComponent(), t.key.userId)
                ssp.getBadgedActivityIcon(info, t.key.userId)
              mIconCache.put(t.key, cachedIcon)
            thumbnailData
              ssp.getTaskThumbnail

这个线程里加载图标是优先使用缓存里的数据,缩略图则是从系统获取。

4. Android缓存策略——LruCache

一般来说,缓存策略主要包含缓存的添加、获取和删除这三类操作。如何添加和获取缓存这个比较好理解,那么为什么还要删除缓存呢?这是因为不管是内存缓存还是硬盘缓存,它们的缓存大小都是有限的。当缓存满了之后,再想其添加缓存,这个时候就需要删除一些旧的缓存并添加新的缓存。

因此LRU(Least Recently Used)缓存算法便应运而生,LRU是近期最少使用的算法,它的核心思想是当缓存满时,会优先淘汰那些近期最少使用的缓存对象。采用LRU算法的缓存有两种:LrhCache和DisLruCache,分别用于实现内存缓存和硬盘缓存,其核心思想都是LRU缓存算法。

image

引用:https://www.jianshu.com/p/b49a111147ee 彻底解析Android缓存机制——LruCache

你可能感兴趣的:(最近任务图标,缩略图的加载和缓存)