美文网首页
最近任务图标,缩略图的加载和缓存

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

作者: shuihuba | 来源:发表于2018-10-26 13:13 被阅读0次

    简述

    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

    相关文章

      网友评论

          本文标题:最近任务图标,缩略图的加载和缓存

          本文链接:https://www.haomeiwen.com/subject/abfctqtx.html