Fragment 可见性监听通过androidx 运行,多种 case 完美兼容!(1)

总结

我最近从朋友那里收集到了2020-2021BAT 面试真题解析,内容很多也很系统,包含了很多内容:Android 基础、Java 基础、Android 源码相关分析、常见的一些原理性问题等等,可以很好地帮助大家深刻理解Android相关知识点的原理以及面试相关知识

这份资料把大厂面试中常被问到的技术点整理成了PDF,包知识脉络 + 诸多细节;还有 高级架构技术进阶脑图 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

这里也分享给广大面试同胞们,希望每位程序猿们都能面试成功~

Android 基础知识点

Java 基础知识点

Android 源码相关分析

常见的一些原理性问题

腾讯、字节跳动、阿里、百度等BAT大厂 2019-2020面试真题解析

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

}

}

AndroidX 的适配(也是一个坑)

在 AndroidX 当中,FragmentAdapter 和FragmentStatePagerAdapter 的构造方法,添加一个 behavior 参数实现的。

如果我们指定不同的 behavior,会有不同的表现。

1、当 behavior 为BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT 时,ViewPager中切换 Fragment,setUserVisibleHint 方法将不再被调用,他会确保onResume 的正确调用时机。

2、当 behavior 为 BEHAVIOR_SET_USER_VISIBLE_HINT,跟之前的方式是一致的,我们可以通过 setUserVisibleHint 结合 fragment 的生命周期来监听。

//FragmentStatePagerAdapter构造方法

public FragmentStatePagerAdapter(@NonNull FragmentManager fm,

@Behavior int behavior) {

mFragmentManager = fm;

mBehavior = behavior;

}

//FragmentPagerAdapter构造方法

public FragmentPagerAdapter(@NonNull FragmentManager fm,

@Behavior int behavior) {

mFragmentManager = fm;

mBehavior = behavior;

}

@IntDef({BEHAVIOR_SET_USER_VISIBLE_HINT, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT})

private @interface Behavior { }

既然是这样,我们就很好适配呢,直接在 onResume 中调用 checkVisibility方法,判断当前 Fragment 是否可见。

回过头,Behavior 是如何实现的呢?

以 FragmentStatePagerAdapter 为例,我们一起来看看源码。

@SuppressWarnings({“ReferenceEquality”, “deprecation”})

@Override

public void setPrimaryItem(@NonNull ViewGroup container, int position, @NonNull Object object) {

Fragment fragment = (Fragment)object;

if (fragment != mCurrentPrimaryItem) {

if (mCurrentPrimaryItem != null) {

//当前显示Fragment

mCurrentPrimaryItem.setMenuVisibility(false);

if (mBehavior == BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {

if (mCurTransaction == null) {

mCurTransaction = mFragmentManager.beginTransaction();

}

//最大生命周期设置为STARTED,生命周期回退到onPause

mCurTransaction.setMaxLifecycle(mCurrentPrimaryItem, Lifecycle.State.STARTED);

} else {

//可见性设置为false

mCurrentPrimaryItem.setUserVisibleHint(false);

}

}

//将要显示的Fragment

fragment.setMenuVisibility(true);

if (mBehavior == BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {

if (mCurTransaction == null) {

mCurTransaction = mFragmentManager.beginTransaction();

}

//最大 生命周期设置为RESUMED

mCurTransaction.setMaxLifecycle(fragment, Lifecycle.State.RESUMED);

} else {

//可见性设置为true

fragment.se tUserVisibleHint(true);

}

//赋值

mCurrentPrimaryItem = fragment;

}

}

代码比较简单很好理解。

当 mBehavior 设置为BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT 会通过setMaxLifecycle 来修改当前Fragment和将要显示的Fragment的状态,使得只有正在显示的 Fragment执行到 onResume() 方法,其他Fragment 只会执行到 onStart() 方法,并且当 Fragment 切换到不显示状态时触发 onPause() 方法。

当 mBehavior 设置为 BEHAVIOR_SET_USER_VISIBLE_HINT 时,会当 frament 可见性发生变化时调用 setUserVisibleHint() ,也就是跟我们上面提到的第一种懒加载实现方式一样。

4.宿主 Fragment 再嵌套 Fragment

===================================================================================

这种 case 也是比较常见的,比如 ViewPager 嵌套 ViewPager,再嵌套Fragment。

宿主Fragment在生命周期执行的时候会相应的分发到子Fragment中,但是setUserVisibleHint和onHiddenChanged却没有进行相应的回调。试想一下,一个ViewPager中有一个FragmentA的tab,而FragmentA中有一个子FragmentB,FragmentA被滑走了,FragmentB并不能接收到setUserVisibleHint事件,onHiddenChange事件也是一样的。

那有没有办法监听到宿主的 setUserVisibleHint 和 ,onHiddenChange事件呢?

方法肯定是有的。

1、第一种方法,宿主 Fragment 提供可见性的回调,子 Fragment 监听改回调,有点类似于观察者模式。难点在于子 Fragment 要怎么拿到宿主Fragment。

2、第二种 case,宿主 Fragment 可见性变化的时候,主动去遍历所有的 的Fragment,调用 子 Fragment 的相应方法。

第一种方法


是这样的,宿主 Fragment 提供可见性的回调,子 Fragment 监听改回调,有点类似于观察者模式。也有点类似于 Rxjava 中下游持有。

第一步 我们先定义一个接口。

interface OnFragmentVisibilityChangedListener {

fun onFragmentVisibilityChanged(visible: Boolean)

}

第二步 在 BaseVisibilityFragment 中提供addOnVisibilityChangedListener 和removeOnVisibilityChangedListener 方法,这里需要注意的是,我们需要用一个 ArrayList 来保存所有的 listener,因为一个宿主 Fragment 可能有多个子 Fragment。

当 Fragment 可见性变化的时候,会遍历 List 调用OnFragmentVisibilityChangedListener 的onFragmentVisibilityChanged 方法 。

open class BaseVisibilityFragment : Fragment(), View.OnAttachStateChangeListener,

OnFragmentVisibilityChangedListener {

private val listeners = ArrayList()

fun addOnVisibilityChangedListener(listener: OnFragmentVisibilityChangedListener?) {

listener?.apply {

listeners.add(this)

}

}

fun removeOnVisibilityChangedListener(listener: OnFragmentVisibilityChangedListener?) {

listener?.apply {

listeners.remove(this)

}

}

private fun checkVisibility(expected: Boolean) {

if (expected == visible) return

val parentVisible =

if (localParentFragment == null) parentActivityVisible

else localParentFragment?.isFragmentVisible() ?: false

val superVisible = super.isVisible()

val hintVisible = userVisibleHint

val visible = parentVisible && superVisible && hintVisible

if (visible != this.visible) {

this.visible = visible

listeners.forEach { it ->

it.onFragmentVisibilityChanged(visible)

}

onVisibilityChanged(this.visible)

}

}

第三步 在 Fragment attach 的时候,我们通过 getParentFragment 方法,拿到宿主 Fragment,进行监听。这样,当宿主 Fragment 可见性变化的时候,子 Fragment 能感应到。

override fun onAttach(context: Context) {

super.onAttach(context)

val parentFragment = parentFragment

if (parentFragment != null && parentFragment is BaseVisibilityFragment) {

this.localParentFragment = parentFragment

info(“onAttach, localParentFragment is $localParentFragment”)

localParentFragment?.addOnVisibilityChangedListener(this)

}

checkVisibility(true)

}

第二种方法


第二种方法,它的实现思路是这样的,宿主 Fragment 生命周期发生变化的时候,遍历子 Fragment,调用相应的方法,通知生命周期发生变化。

//当自己的显示隐藏状态改变时,调用这个方法通知子Fragment

private void notifyChildHiddenChange(boolean hidden) {

if (isDetached() || !isAdded()) {

return;

}

FragmentManager fragmentManager = getChildFragmentManager();

List fragments = fragmentManager.getFragments();

if (fragments == null || fragments.isEmpty()) {

return;

}

for (Fragment fragment : fragments) {

if (!(fragment instanceof IPareVisibilityObserver)) {

continue;

}

((IPareVisibilityObserver) fragment).onParentFragmentHiddenChanged(hidden);

}

}

5完整代码

==============================================================

/**

  • Created by jun xu on 2020/11/26.

*/

interface OnFragmentVisibilityChangedListener {

fun onFragmentVisibilityChanged(visible: Boolean)

}

/**

  • Created by jun xu on 2020/11/26.

  • 支持以下四种 case

    1. 支持 viewPager 嵌套 fragment,主要是通过 setUserVisibleHint 兼容,
  • FragmentStatePagerAdapter BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT 的 case,因为这时候不会调用 setUserVisibleHint 方法,在 onResume check 可以兼容

    1. 直接 fragment 直接 add, hide 主要是通过 onHiddenChanged
    1. 直接 fragment 直接 replace ,主要是在 onResume 做判断
    1. Fragment 里面用 ViewPager, ViewPager 里面有多个 Fragment 的,通过 setOnVisibilityChangedListener 兼容,前提是一级 Fragment 和 二级 Fragment 都必须继承 BaseVisibilityFragment, 且必须用 FragmentPagerAdapter 或者 FragmentStatePagerAdapter
  • 项目当中一级 ViewPager adapter 比较特殊,不是 FragmentPagerAdapter,也不是 FragmentStatePagerAdapter,导致这种方式用不了

*/

open class BaseVisibilityFragment : Fragment(), View.OnAttachStateChangeListener,

OnFragmentVisibilityChangedListener {

companion object {

const val TAG = “BaseVisibilityFragment”

}

/**

  • ParentActivity是否可见

*/

private var parentActivityVisible = false

/**

  • 是否可见(Activity处于前台、Tab被选中、Fragment被添加、Fragment没有隐藏、Fragment.View已经Attach)

*/

private var visible = false

private var localParentFragment: BaseVisibilityFragment? =

null

private val listeners = ArrayList()

fun addOnVisibilityChangedListener(listener: OnFragmentVisibilityChangedListener?) {

listener?.apply {

listeners.add(this)

}

}

fun removeOnVisibilityChangedListener(listener: OnFragmentVisibilityChangedListener?) {

listener?.apply {

listeners.remove(this)

}

}

override fun onAttach(context: Context) {

info(“onAttach”)

super.onAttach(context)

val parentFragment = parentFragment

if (parentFragment != null && parentFragment is BaseVisibilityFragment) {

this.localParentFragment = parentFragment

localParentFragment?.addOnVisibilityChangedListener(this)

}

checkVisibility(true)

}

override fun onDetach() {

info(“onDetach”)

localParentFragment?.removeOnVisibilityChangedListener(this)

super.onDetach()

checkVisibility(false)

localParentFragment = null

}

override fun onResume() {

info(“onResume”)

super.onResume()

onActivityVisibilityChanged(true)

}

override fun onPause() {

info(“onPause”)

super.onPause()

onActivityVisibilityChanged(false)

}

/**

  • ParentActivity可见性改变

*/

protected fun onActivityVisibilityChanged(visible: Boolean) {

parentActivityVisible = visible

checkVisibility(visible)

}

/**

  • ParentFragment可见性改变

*/

override fun onFragmentVisibilityChanged(visible: Boolean) {

checkVisibility(visible)

}

总结

开发是面向对象。我们找工作应该更多是面向面试。哪怕进大厂真的只是去宁螺丝,但你要进去得先学会面试的时候造飞机不是么?

作者13年java转Android开发,在小厂待过,也去过华为,OPPO等,去年四月份进了阿里一直到现在。等大厂待过也面试过很多人。深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

相信它会给大家带来很多收获:

960页全网最全Android开发笔记

资料太多,全部展示会影响篇幅,暂时就先列举这些部分截图

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

中级Android工程师,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

相信它会给大家带来很多收获:

[外链图片转存中…(img-QyHykwrm-1715266026561)]

[外链图片转存中…(img-xRBb7gyg-1715266026561)]

资料太多,全部展示会影响篇幅,暂时就先列举这些部分截图

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值