網(wǎng)站首頁(yè) 編程語言 正文
前言:
TabLayout+ViewPager+Fragment是我們開發(fā)常用的組合。ViewPager的默認(rèn)機(jī)制就是把全部的Fragment都加載出來,而為了保障一些用戶體驗(yàn),我們使用懶加載的Fragment,就是讓我們?cè)儆脩艨梢娺@個(gè)Fragment之后才處理業(yè)務(wù)邏輯。
而我們?cè)谝恍┰O(shè)備或版本中可能就出現(xiàn)懶加載失效的問題。其實(shí)谷歌早就把一些懶加載的方案都標(biāo)記棄用了,我們一直都用的老的隨時(shí)會(huì)失效的Api。萬一哪天徹底失效了就會(huì)導(dǎo)致線上事故。
接下來我們就看看Fragment的懶加載是如何演變的。谷歌又是推薦我們?nèi)绾问褂玫摹?/p>
1. Support時(shí)代的懶加載
在AndroidX還沒出來的時(shí)候,大家的懶加載應(yīng)該都是這樣。判斷setUserVisibleHint的方法,當(dāng)用戶可見的時(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//是否第一次可見 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依賴注入獲取 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,重寫次方法 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, 但還沒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ì)用戶是否可見 * @param fragment 要判斷的fragment * @return true表示對(duì)用戶可見 */ 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)定,我也遇到過不回調(diào)的時(shí)候。
Android出來之后,給?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)聽 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出來之后。我們的?FragmentStatePagerAdapter
?退出歷史舞臺(tái)。
即便能用,即便效果還是和ViewPage2的效果一樣,但是還是標(biāo)記廢棄了。具體原因我也不知道,據(jù)說是因?yàn)槔习姹緯?huì)出現(xiàn)問題導(dǎo)致數(shù)據(jù)丟失,頁(yè)面空白。
ViewPage2我們都知道內(nèi)部是通過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的話,可以無縫切換過來。
原文鏈接:https://juejin.cn/post/7101195320318492702
相關(guān)推薦
- 2022-04-12 網(wǎng)絡(luò)編程——Http請(qǐng)求方式Get與Post
- 2022-05-05 輕量級(jí)ORM框架Dapper應(yīng)用之實(shí)現(xiàn)In操作_實(shí)用技巧
- 2022-08-01 OpenCV根據(jù)面積篩選連通域?qū)W習(xí)示例_python
- 2023-02-17 Python排序算法之堆排序算法_python
- 2022-11-09 GO?語言運(yùn)行環(huán)境的基礎(chǔ)知識(shí)_Golang
- 2022-12-11 React?state狀態(tài)屬性用法講解_React
- 2022-08-31 C++?OpenCV裁剪圖片時(shí)發(fā)生報(bào)錯(cuò)的解決方式_C 語言
- 2022-10-10 pycharm創(chuàng)建并使用虛擬環(huán)境的詳細(xì)圖文教程_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支