Android開(kāi)發(fā)之Fragment懶加載的幾種方式及性能對(duì)比
前言:
TabLayout+ViewPager+Fragment是我們開(kāi)發(fā)常用的組合。ViewPager的默認(rèn)機(jī)制就是把全部的Fragment都加載出來(lái),而為了保障一些用戶(hù)體驗(yàn),我們使用懶加載的Fragment,就是讓我們?cè)儆脩?hù)可見(jiàn)這個(gè)Fragment之后才處理業(yè)務(wù)邏輯。
而我們?cè)谝恍┰O(shè)備或版本中可能就出現(xiàn)懶加載失效的問(wèn)題。其實(shí)谷歌早就把一些懶加載的方案都標(biāo)記棄用了,我們一直都用的老的隨時(shí)會(huì)失效的Api。萬(wàn)一哪天徹底失效了就會(huì)導(dǎo)致線上事故。
接下來(lái)我們就看看Fragment的懶加載是如何演變的。谷歌又是推薦我們?nèi)绾问褂玫摹?/p>
1. Support時(shí)代的懶加載
在AndroidX還沒(méi)出來(lái)的時(shí)候,大家的懶加載應(yīng)該都是這樣。判斷setUserVisibleHint的方法,當(dāng)用戶(hù)可見(jiàn)的時(shí)候才回調(diào)方法去加載邏輯。
例如的我封裝:
abstract class BaseVDBLazyLoadingFragment<VM : BaseViewModel, VDB : ViewDataBinding> : AbsFragment() {
protected lateinit var mViewModel: VM
protected lateinit var mBinding: VDB
private var isViewCreated = false//布局是否被創(chuàng)建
private var isLoadData = false//數(shù)據(jù)是否加載
private var isFirstVisible = true//是否第一次可見(jiàn)
protected lateinit var mGLoadingHolder: Gloading.Holder
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
isViewCreated = true
init()
startObserve()
}
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
if (isFragmentVisible(this) && this.isAdded) {
if (parentFragment == null || isFragmentVisible(parentFragment)) {
onLazyInitData()
isLoadData = true
if (isFirstVisible) isFirstVisible = false
}
}
}
//使用這個(gè)方法簡(jiǎn)化ViewModewl的Hilt依賴(lài)注入獲取
protected inline fun <reified VM : BaseViewModel> getViewModel(): VM {
val viewModel: VM by viewModels()
return viewModel
}
//反射獲取ViewModel實(shí)例
private fun createViewModel(): VM {
return ViewModelProvider(this).get(getVMCls(this))
}
override fun setContentView(container: ViewGroup?): View {
mViewModel = createViewModel()
//觀察網(wǎng)絡(luò)數(shù)據(jù)狀態(tài)
mViewModel.getActionLiveData().observe(viewLifecycleOwner, stateObserver)
val config = getDataBindingConfig()
mBinding = DataBindingUtil.inflate(layoutInflater, config.getLayout(), container, false)
mBinding.lifecycleOwner = viewLifecycleOwner
if (config.getVmVariableId() != 0) {
mBinding.setVariable(
config.getVmVariableId(),
config.getViewModel()
)
}
val bindingParams = config.getBindingParams()
bindingParams.forEach { key, value ->
mBinding.setVariable(key, value)
}
return mBinding.root
}
abstract fun getDataBindingConfig(): DataBindingConfig
abstract fun startObserve()
abstract fun init()
abstract fun onLazyInitData()
//Loading Create Root View
override fun transformRootView(view: View): View {
mGLoadingHolder = generateGLoading(view)
return mGLoadingHolder.wrapper
}
//如果要替換GLoading,重寫(xiě)次方法
open protected fun generateGLoading(view: View): Gloading.Holder {
return Gloading.getDefault().wrap(view).withRetry {
onGoadingRetry()
}
}
protected open fun onGoadingRetry() {
}
override fun onNetworkConnectionChanged(isConnected: Boolean, networkType: NetWorkUtil.NetworkType?) {
}
// ============================ Lazy Load begin ↓ =============================
override fun setUserVisibleHint(isVisibleToUser: Boolean) {
super.setUserVisibleHint(isVisibleToUser)
if (isFragmentVisible(this) && !isLoadData && isViewCreated && this.isAdded) {
onLazyInitData()
isLoadData = true
}
}
override fun onHiddenChanged(hidden: Boolean) {
super.onHiddenChanged(hidden)
//onHiddenChanged調(diào)用在Resumed之前,所以此時(shí)可能fragment被add, 但還沒(méi)resumed
if (!hidden && !this.isResumed)
return
//使用hide和show時(shí),fragment的所有生命周期方法都不會(huì)調(diào)用,除了onHiddenChanged()
if (!hidden && isFirstVisible && this.isAdded) {
onLazyInitData()
isFirstVisible = false
}
}
override fun onDestroy() {
super.onDestroy()
isViewCreated = false
isLoadData = false
isFirstVisible = true
}
/**
* 當(dāng)前Fragment是否對(duì)用戶(hù)是否可見(jiàn)
* @param fragment 要判斷的fragment
* @return true表示對(duì)用戶(hù)可見(jiàn)
*/
private fun isFragmentVisible(fragment: Fragment?): Boolean {
return !fragment?.isHidden!! && fragment.userVisibleHint
}
}使用的示例:
mBinding.viewPager.bindFragment(
supportFragmentManager,
listOf(LazyLoad1Fragment.obtainFragment(), LazyLoad2Fragment.obtainFragment(), LazyLoad3Fragment.obtainFragment()),
listOf("Demo1", "Demo2", "Demo3")
)
mBinding.tabLayout.setupWithViewPager(mBinding.viewPager)擴(kuò)展方法:
fun ViewPager.bindFragment(
fm: FragmentManager,
fragments: List<Fragment>,
pageTitles: List<String>? = null,
behavior: Int = 0
): ViewPager {
offscreenPageLimit = fragments.size - 1
adapter = object : FragmentStatePagerAdapter(fm, behavior) {
override fun getItem(p: Int) = fragments[p]
override fun getCount() = fragments.size
override fun getPageTitle(p: Int) = if (pageTitles == null) null else pageTitles[p]
}
return this
}Fragment:
class LazyLoad1Fragment : BaseVDBLazyLoadingFragment<EmptyViewModel, FragmentDemo2Binding>() {
companion object {
fun obtainFragment(): LazyLoad1Fragment {
return LazyLoad1Fragment()
}
}
override fun getDataBindingConfig(): DataBindingConfig {
return DataBindingConfig(R.layout.fragment_demo2)
}
override fun startObserve() {
}
override fun init() {
YYLogUtils.w("LazyLoad1Fragment - init")
mBinding.tvPage2.click {
Demo2Pager2Activity.startInstance()
}
}
//重新生成GLoading對(duì)象
override fun generateGLoading(view: View): Gloading.Holder {
return Gloading.from(GloadingRoatingAdapter()).wrap(view).withRetry {
onGoadingRetry()
}
}
override fun onResume() {
super.onResume()
YYLogUtils.w("LazyLoad1Fragment - onResume")
}
override fun onGoadingRetry() {
toast("重試一個(gè)請(qǐng)求")
onLazyInitData()
}
override fun onLazyInitData() {
YYLogUtils.w("LazyLoad1Fragment - initData")
//模擬的Loading的情況
showStateLoading()
CommUtils.getHandler().postDelayed({
showStateSuccess()
}, 2500)
}
}到此就實(shí)現(xiàn)了onLazyInitData的回調(diào),只有出現(xiàn)Fragment顯示在前臺(tái)的時(shí)候才會(huì)調(diào)用方法,執(zhí)行邏輯。

2. AndrodX時(shí)代的懶加載
每次判斷 setUserVisibleHint 和 onHiddenChanged 也麻煩,并且他們并不穩(wěn)定,我也遇到過(guò)不回調(diào)的時(shí)候。
Android出來(lái)之后,給 FragmentStatePagerAdapter 添加了一個(gè) @Behavior int behavior 的參數(shù)。
其本質(zhì)就是內(nèi)部幫你處理和切換MaxLifecycle:
mCurTransaction.setMaxLifecycle(fragment, Lifecycle.State.STARTED);
mCurTransaction.setMaxLifecycle(fragment, Lifecycle.State.RESUMED);
如何使用呢:
mBinding.viewPager.bindFragment(
supportFragmentManager,
listOf(LazyLoad1Fragment.obtainFragment(), LazyLoad2Fragment.obtainFragment(), LazyLoad3Fragment.obtainFragment()),
listOf("Demo1", "Demo2", "Demo3"),
behavior = 1
)之前的擴(kuò)展方法以及預(yù)留了 behavior 參數(shù),當(dāng)為1的時(shí)候就不會(huì)回調(diào) setUserVisibleHint 方法了,我們直接監(jiān)聽(tīng) OnResume 即可。
class LazyLoad3Fragment : BaseVDBLoadingFragment<EmptyViewModel, FragmentDemo2Binding>() {
var isLoaded = false
companion object {
fun obtainFragment(): LazyLoad3Fragment {
return LazyLoad3Fragment()
}
}
override fun getDataBindingConfig(): DataBindingConfig {
return DataBindingConfig(R.layout.fragment_demo2)
}
//重新生成GLoading對(duì)象
override fun generateGLoading(view: View): Gloading.Holder {
return Gloading.from(GloadingLoadingAdapter()).wrap(view).withRetry {
onGoadingRetry()
}
}
override fun startObserve() {
}
override fun init() {
YYLogUtils.w("LazyLoad3Fragment - init")
}
private fun initData() {
YYLogUtils.w("LazyLoad3Fragment - initData")
//模擬的Loading的情況
showStateLoading()
CommUtils.getHandler().postDelayed({
showStateSuccess()
}, 2500)
isLoaded = true
}
override fun onResume() {
super.onResume()
YYLogUtils.w("LazyLoad3Fragment - onResume")
if (!isLoaded) initData()
}
override fun onGoadingRetry() {
toast("重試一個(gè)請(qǐng)求")
initData()
}
}注意這個(gè)頁(yè)面繼承的就不是我們自定義的懶加載Fragment了。普通的Fragment 回調(diào) onResume 即可。
3. ViewPager2時(shí)代的懶加載
ViewPager2出來(lái)之后。我們的 FragmentStatePagerAdapter 退出歷史舞臺(tái)。

即便能用,即便效果還是和ViewPage2的效果一樣,但是還是標(biāo)記廢棄了。具體原因我也不知道,據(jù)說(shuō)是因?yàn)槔习姹緯?huì)出現(xiàn)問(wèn)題導(dǎo)致數(shù)據(jù)丟失,頁(yè)面空白。
ViewPage2我們都知道內(nèi)部是通過(guò)RV實(shí)現(xiàn)的。但是對(duì)于Fragment的處理有單獨(dú)的Adapter實(shí)現(xiàn)。
擴(kuò)展方法:
/**
* 給ViewPager2綁定Fragment
*/
fun ViewPager2.bindFragment(
fm: FragmentManager,
lifecycle: Lifecycle,
fragments: List<Fragment>
): ViewPager2 {
offscreenPageLimit = fragments.size - 1
adapter = object : FragmentStateAdapter(fm, lifecycle) {
override fun getItemCount(): Int = fragments.size
override fun createFragment(position: Int): Fragment = fragments[position]
}
return this
}使用:
mBinding.viewPager2.bindFragment(
supportFragmentManager,
this.lifecycle,
listOf(LazyLoad1Fragment.obtainFragment(), LazyLoad2Fragment.obtainFragment(), LazyLoad3Fragment.obtainFragment())
)
val title = listOf("Demo1", "Demo2", "Demo3")
TabLayoutMediator(mBinding.tabLayout, mBinding.viewPager2) { tab, position ->
//回調(diào)
tab.text = title[position]
}.attach()使用的方式和ViewPager差不多,這里的Fragment也是使用普通的Fragment即可。
4. ViewPage和ViewPager2的性能對(duì)比
內(nèi)存占用分別取三組數(shù)據(jù)
ViewPager數(shù)據(jù)

一。111 二。117.6 三。115.1
ViewPager2數(shù)據(jù)

一。110 二。107.4 三。107.6
結(jié)論 ViewPager2基于RV實(shí)現(xiàn)的效果還是比老版ViewPager要騷好一點(diǎn)。
并且老版本標(biāo)記廢棄,大家如果是用ViewPager2的話,還是推薦使用ViewPager2實(shí)現(xiàn)。如果大家還是用的老版本的ViewPager也推薦使用behavor參數(shù)。使用 onResume 實(shí)現(xiàn)懶加載的實(shí)現(xiàn)。以后再換到ViewPager2的話,可以無(wú)縫切換過(guò)來(lái)。
到此這篇關(guān)于Android開(kāi)發(fā)之Fragment懶加載的幾種方式及性能對(duì)比的文章就介紹到這了,更多相關(guān)Android Fragment懶加載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android實(shí)現(xiàn)設(shè)置APP灰白模式效果
大家好,本篇文章主要講的是Android實(shí)現(xiàn)設(shè)置APP灰白模式效果,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12
Android依據(jù)名字通過(guò)反射獲取在drawable中的圖片
依據(jù)圖片的名字,通過(guò)反射獲取其在drawable中的ID,在根據(jù)此ID顯示圖片,具體實(shí)現(xiàn)如下,感興趣的朋友可以參考下哈2013-06-06
Android如何幫助用戶(hù)自動(dòng)接聽(tīng)或者掛斷來(lái)電
這篇文章主要為大家詳細(xì)介紹了Android幫助用戶(hù)自動(dòng)接聽(tīng)或者掛斷來(lái)電,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05
Android ApplicationContext接口深入分析
ApplicationContext是Spring應(yīng)用程序中的中央接口,由于繼承了多個(gè)組件,使得ApplicationContext擁有了許多Spring的核心功能,如獲取bean組件,注冊(cè)監(jiān)聽(tīng)事件,加載資源文件等2022-11-11
Android優(yōu)雅地處理按鈕重復(fù)點(diǎn)擊的幾種方法
這篇文章主要介紹了Android優(yōu)雅地處理按鈕重復(fù)點(diǎn)擊的幾種方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-09-09
淺談Android中Drawable使用知識(shí)總結(jié)
本篇文章主要介紹了淺談Android中Drawable使用知識(shí)總結(jié),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
android手機(jī)端與PC端使用adb forword通信
這篇文章主要介紹了android手機(jī)端與PC端使用adb forword通信的相關(guān)資料,需要的朋友可以參考下2017-04-04

