离屏预加载|性能优化
安卓原生项目中,性能优化在 可用性、流畅性和稳定性等质量方面,用于决定是否可提高用户留存率、转化率。
单拎出 流畅性方面,我们的目的是减少使用中的卡顿、响应时间久等问题,给与用户一个操作流畅的体验。由此可针对 启动速度、页面显示速度、响应速度3个方面进行优化。
这里介绍其中一种优化方案
提高页面显示速度方案——离屏预加载
设计实现
问题: ActivityA跳转ActivityB,ActivityB页面布局较复杂,显示速度有待提升。
优化: ActivityA跳转到ActivityB之前,预加载ActivityB页面布局并缓存。等待到进入ActivityA时添加已缓存的布局并进行渲染显示。(且一些数据资源也可以在ActivityA预先获得并缓存后使用)
- 预加载实现
将对将跳转Activity页面布局预加载执行放在UI线程空闲时段执行。
抽象类BasePreloadView继承自FrameLayout,其中实现缓存方案、LayoutInflate能力。
预定义[实现]类PreloadBundleView,即包装类(目前未做扩展),实现自BasePreloadView。其实这里想要仿照LifecycleRegisty中的addObserver$ObserverWithState方法,来实现预加载的设计。鉴于当前预加载实现无需那么复杂,所以简化为直接以预定义[实现]类实现。
object ActivityPreloadManager {
/** View预加载 */
fun preload(intent: Intent, context: Context, resId: Int) {
// 执行缓存时,须判断缓存中不存在
val preloadView:BasePreloadView? = BasePreloadView.getVal(intent)
if (preloadView != null) {
throw IllegalStateException("Error:布局View重复加载..." )
}
val clazz:Class<PreloadBundleView> = PreloadBundleView::class.java
Looper.myQueue().addIdleHandler {
// 不占用UI线程,空闲时执行, 反射获取布局实例并缓存
val instance = clazz.getConstructor(Context::class.java).newInstance(context)
instance.onCreatedInit(resId)
// 缓存
BasePreloadView.putVal(intent, instance)
true
}
}
}
- 添加已加载布局View到ActivityB
定义活动页面基类,并自动完成预加载布局的添加渲染显示逻辑。
在跳转到的新页面生命周期方法onCreate中。若缓存已存在,则获取并添加到当前活动页面DecorView中。若不存在(兜底方案)则通过KClass反射获取预定义[实现]类PreloadBundleView实例,执行inflate布局、缓存并添加到当前页面DecorView逻辑。以完成当前页面渲染显示。
/** Activity基类AppCompatActivity可换。进入Activity之前布局未预加载时,在onCreate执行重新加载。 */
abstract class BasePreloadActivity: AppCompatActivity() {
private lateinit var preloadBaseView: PreloadBundleView
// KClass = Context::class
// Class = Context::class.java 泛型类能力拓展
private fun preloadViewClassReflect(): KClass<PreloadBundleView> = PreloadBundleView::class
abstract fun getResId(): Int
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
preloadBaseView = preloadViewClassReflect().createView(intent, this, getResId())
setContentView(preloadBaseView) // 添加自定义ViewGroup到Activity
}
}
设计亮点,上面Lib实现预留了可改造设计的后手,即 将预定义的[实现]类以泛型形式设计,可实现它的通用性设计思想。这里做为开端,更优秀、实用设计可自行fork后尝试实现。
该离屏预加载的moduleLib实现源码,以及使用可在开源的项目源码中进行查看。