先看下Demo实现的效果吧。你们对这种效果必定不陌生,知乎,掘金等app都用到了这种效果。 git
viewpager
结合
tablayout
时,多个
fragment
在进行数据加载的时候,当且仅当
fragment
对用户可见的时候,才进行数据加载,这里的数据加载能够是网络请求,数据库请求,是须要消耗资源的,若是不使用懒加载,对于那些用户未打开的页面,因为
viewpager
的加载机制,即便用户未打开的
fragment
也有可能进行数据加载,形成资源白白的浪费,所以为了良好的用户体验,懒加载是有必要的。
fragment
懒加载实现的关键在于其的setUserVisibleHint(isVisibleToUser: Boolean)
方法,该方法在fragment
对用户由可见变为不可见以及由不可见变为可见时都会回调。咱们建立抽象AbstractLazyInitFrag
,对其进行封装。首先咱们引入isVisibleToUser
变量,负责保存当前fragment
对用户的可见状态。同时还有几个值得注意的地方:github
setUserVisibleHint(isVisibleToUser: Boolean)
方法的回调时机并无与fragment
的生命周期有确切的关联,好比说,回调时机有可能在onCreateView
方法以后,也可能在onCreateView
方法以前。所以,必须引入一个标志位isPrepareView
判断view
是否建立完成,否则,很容易会形成空指针异常。咱们初始化该变量为false,在onViewCreated
中,也就是view
建立完成后,将其赋值为true
。代码中是这样:override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
isPrepareView = true
}
复制代码
isInitData
,初始为false,在数据加载完成以后,将其赋值为true
。至此,咱们的懒加载方法考虑了全部条件。也就是当isVisibleToUser
为true
,isInitData
为false
,isPrepareView
为true
时,进行数据加载,而且加载后为了防止重复调用,将isInitData
赋值为true
。代码以下:private fun lazyInitData() {
if (!isInitData && isVisibleToUser && isPrepareView) {
isInitData = true
initData()
}
}
复制代码
其中initData()
为抽象方法,由子类实现,在这里操做数据加载的逻辑。数据库
setUserVisibleHint(isVisibleToUser: Boolean)
方法中是必须调用的。代码以下:/*当fragment由可见变为不可见和不可见变为可见时回调*/
override fun setUserVisibleHint(isVisibleToUser: Boolean) {
this.isVisibleToUser = isVisibleToUser //标志位保存fragment对用户的可见状态
lazyInitData()
}
复制代码
其次,很容易忽略的一点。对于上图中第一个fragment
,若是setUserVisibleHint(isVisibleToUser: Boolean)
方法在onCreateView
以前调用的话,若是懒加载方法只在setUserVisibleHint(isVisibleToUser: Boolean)
中调用,那么该fragment将只能在被主动切换一次以后才能加载数据,这确定是不可能的,所以,咱们须要在view
建立完成以后,也进行一次调用。思来想去,在onActivityCreated
方法中是最合适的。咱们在继承的时候,在onViewCreated
方法中进行一些初始化就好了,这样不会引发冲突。代码以下:bash
/*fragment生命周期中onViewCreated以后的方法 在这里调用一次懒加载 避免第一次可见不加载数据*/
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
lazyInitData()
}
复制代码
贴上完整的封装好的抽象基类fragment:网络
abstract class AbstractLazyInitFrag : Fragment() {
private var isInitData = false /*标志位 判断数据是否初始化*/
private var isVisibleToUser = false /*标志位 判断fragment是否可见*/
private var isPrepareView = false /*标志位 判断view已经加载完成 避免空指针操做*/
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return inflater.inflate(getLayoutId(), container, false)
}
abstract fun getLayoutId(): Int
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
isPrepareView = true
}
/*加载数据的方法,由子类实现*/
abstract fun initData()
/*懒加载方法*/
private fun lazyInitData() {
if (!isInitData && isVisibleToUser && isPrepareView) {
isInitData = true
initData()
}
}
/*当fragment由可见变为不可见和不可见变为可见时回调*/
override fun setUserVisibleHint(isVisibleToUser: Boolean) {
this.isVisibleToUser = isVisibleToUser
lazyInitData()
}
/*fragment生命周期中onViewCreated以后的方法 在这里调用一次懒加载 避免第一次可见不加载数据*/
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
lazyInitData()
}
}
复制代码
顺便贴下Demo中测试的继承的子类:app
class FragLazyInitTest : AbstractLazyInitFrag() {
private val dataList = ArrayList<String>()
private val adapter = ListAdapter(dataList)
/*初始化通常在这里实现*/
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
recyclerView.layoutManager = LinearLayoutManager(context)
recyclerView.adapter = adapter
}
override fun getLayoutId(): Int {
return R.layout.frag_lazy_init_test
}
override fun initData() {
/*模拟的加载数据过程,实际场景通常是网络请求或者数据库等耗时操做*/
swipeRefreshLayoutFLIT.isRefreshing = true
swipeRefreshLayoutFLIT.postDelayed({
swipeRefreshLayoutFLIT.isRefreshing = false
dataList.add("data1")
dataList.add("data2")
dataList.add("data3")
dataList.add("data4")
dataList.add("data5")
dataList.add("data6")
dataList.add("data7")
dataList.add("data8")
adapter.notifyDataSetChanged()
}, 2000)
}
}
复制代码
这是第一次在掘金上写文章,也几乎是第一次写博客吧。感受描述的很凌乱,你们若是有疑问能够去看看Demo,而后多多交流.post