简述
最近任务卡片上的图片资源主要有应用图标和缩略图。应用图标一般是固定的,加载后就放在缓存里,不需要每次从系统里加载。缩略图则有可能会经常变化,所以经常需要实时加载,加载的时候一般使用异步加载。
图标和缩略图从加载到显示在卡片上的流程可以分为三部分。
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缓存算法。
引用:https://www.jianshu.com/p/b49a111147ee 彻底解析Android缓存机制——LruCache