美文网首页Android开发互联网科技Android开发经验谈
Android:Fragment懒加载的实现以及自己的封装思路

Android:Fragment懒加载的实现以及自己的封装思路

作者: 881ef7b85f62 | 来源:发表于2019-08-20 14:22 被阅读11次

    1.什么是fragment懒加载以及为什么要使用fragment懒加载。

    先看下Demo实现的效果吧。大家对这种效果一定不陌生,知乎,掘金等app都用到了这种效果。

    这里探索的懒加载是当viewpager 结合tablayout时,多个fragment在进行数据加载的时候,当且仅当fragment对用户可见的时候,才进行数据加载,这里的数据加载可以是网络请求,数据库请求,是需要消耗资源的,如果不使用懒加载,对于那些用户未打开的页面,由于viewpager的加载机制,即使用户未打开的fragment也有可能进行数据加载,造成资源白白的浪费,因此为了良好的用户体验,懒加载是有必要的。

    2.fragment懒加载的实现方式以及封装。

    fragment懒加载实现的关键在于其的setUserVisibleHint(isVisibleToUser: Boolean)方法,该方法在fragment对用户由可见变为不可见以及由不可见变为可见时都会回调。我们创建抽象AbstractLazyInitFrag,对其进行封装。首先我们引入isVisibleToUser变量,负责保存当前fragment对用户的可见状态。同时还有几个值得注意的地方:

    1. 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
        }
    
    1. 数据初始化只应该加载一次,因此,引入第二个标志位,isInitData,初始为false,在数据加载完成之后,将其赋值为true。至此,我们的懒加载方法考虑了所有条件。也就是当isVisibleToUsertrueisInitDatafalseisPrepareViewtrue时,进行数据加载,并且加载后为了防止重复调用,将isInitData赋值为true。代码如下:
    private fun lazyInitData() {
            if (!isInitData && isVisibleToUser && isPrepareView) {
                isInitData = true
                initData()
            }
        }
    

    其中initData()为抽象方法,由子类实现,在这里操作数据加载的逻辑。

    1. 该方法的调用时机。首先是 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方法中进行一些初始化就行了,这样不会引起冲突。代码如下:

    /*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中测试的继承的子类:

    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)
        }
    }
    

    3.Demo地址

    github.com/Lesincs/Laz…

    学习分享,共勉

    题外话,我从事Android开发已经五年了,此前我指导过不少同行。但很少跟大家一起探讨,正好最近我花了一个多月的时间整理出来一份包括不限于高级UI、性能优化、移动架构师、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter等全方面的Android进阶实践技术,今天暂且开放给有需要的人,若有关于此方面可以转发+关注+点赞后加群 878873098 领取,或者评论与我一起交流探讨。

    image

    资料免费领取方式:转发+关注+点赞后,加入点击链接加入群聊:Android高级开发交流群(878873098)即可获取免费领取方式!

    重要的事说三遍,关注!关注!关注!

    相关文章

      网友评论

        本文标题:Android:Fragment懒加载的实现以及自己的封装思路

        本文链接:https://www.haomeiwen.com/subject/pukpsctx.html