美文网首页Android进阶之路Android开发经验谈Android技术知识
为什么通过 findFragmentByTag 会获取不到 Fr

为什么通过 findFragmentByTag 会获取不到 Fr

作者: 锦岳 | 来源:发表于2019-09-28 13:48 被阅读0次

许多 App 会有类似于微信底部导航的需求,这种需求的处理有多种方式,而 FragmentTabHost + Fragment 是其中一种比较常见的方式。
只是在实际开发过程中,我们经常会遇到一个问题,那就是在向 FragmentTabHost 中添加 fragment 之后,获取不到向其中添加的 fragment 的实例,这是怎么回事呢?
下面我们通过阅读源码来分析一下其中的原因。
FragmentTabHost 与 Fragment 的关联发生在调用 addTab() 时,我们首先看 addTab() 中的 源码:

        tabSpec.setContent(new FragmentTabHost.DummyTabFactory(this.mContext));
        String tag = tabSpec.getTag();
        FragmentTabHost.TabInfo info = new FragmentTabHost.TabInfo(tag, clss, args);
        if (this.mAttached) {
            info.fragment = this.mFragmentManager.findFragmentByTag(tag);
            if (info.fragment != null && !info.fragment.isDetached()) {
                FragmentTransaction ft = this.mFragmentManager.beginTransaction();
                ft.detach(info.fragment);
                ft.commit();
            }
        }

        this.mTabs.add(info);
        this.addTab(tabSpec);

fragment 实例在 if(this.mAttached) 代码块中被创建。那 mAttached 什么时候会是 true 呢?
找遍 FragmentTabHost 源码,就在一处发现了 mAttached 被设置为 true:

    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        String currentTag = this.getCurrentTabTag();
        FragmentTransaction ft = null;
        int i = 0;

        for(int count = this.mTabs.size(); i < count; ++i) {
            FragmentTabHost.TabInfo tab = (FragmentTabHost.TabInfo)this.mTabs.get(i);
            tab.fragment = this.mFragmentManager.findFragmentByTag(tab.tag);
            if (tab.fragment != null && !tab.fragment.isDetached()) {
                if (tab.tag.equals(currentTag)) {
                    this.mLastTab = tab;
                } else {
                    if (ft == null) {
                        ft = this.mFragmentManager.beginTransaction();
                    }

                    ft.detach(tab.fragment);
                }
            }
        }

        this.mAttached = true;
        ft = this.doTabChanged(currentTag, ft);
        if (ft != null) {
            ft.commit();
            this.mFragmentManager.executePendingTransactions();
        }

    }

通过查询资料获知,此方法是继承自 View ——也就是 FragmentTabHost 的非直接父类——的方法,这个方法会在 Activity 的 onResume() 方法之后执行。
到此处,就真相大白了。
由于 addTab() 方法的执行是在 Activity 的 onCreate() 中,而此时 mAttached 仍然处于 false 的状态位,所以并不会创建 Fragment 实例,也就是在调用 findFragmentByTag() 时会获取到 null。


加两条分割线


在网上还看到一个延时获取 Fragment 实例的解决方案,而这个解决方案的原理是什么呢?答案也在这个 onAttachedToWindow() 方法中。
由于在 addTab() 方法中没有能够将 Fragment 实例化,因此在 onAttachedToWindow() 方法中仍然无法获取到 Fragment 实例,也就是

tab.fragment = this.mFragmentManager.findFragmentByTag(tab.tag);

获取到的 tab.fragment 仍然为null。到了这一步,要获取到 Fragment 实例,就只有创建之这一条路可走了,而创建 Fragment 实例的方法在哪里呢?在一个封装好的内部方法 doTabChanged() 方法中:

    @Nullable
    private FragmentTransaction doTabChanged(@Nullable String tag, @Nullable FragmentTransaction ft) {
        FragmentTabHost.TabInfo newTab = this.getTabInfoForTag(tag);
        if (this.mLastTab != newTab) {
            if (ft == null) {
                ft = this.mFragmentManager.beginTransaction();
            }

            if (this.mLastTab != null && this.mLastTab.fragment != null) {
                ft.detach(this.mLastTab.fragment);
            }

            if (newTab != null) {
                if (newTab.fragment == null) {
                    newTab.fragment = Fragment.instantiate(this.mContext, newTab.clss.getName(), newTab.args);
                    ft.add(this.mContainerId, newTab.fragment, newTab.tag);
                } else {
                    ft.attach(newTab.fragment);
                }
            }

            this.mLastTab = newTab;
        }

        return ft;
    }

这里有一句

newTab.fragment = Fragment.instantiate(this.mContext, newTab.clss.getName(), newTab.args)

初始化一个类实例。
但这跟延时获取实例似乎还是不搭边。
我们回到 onAttachedToWindow() 继续向下看,有这么一句:

this.mFragmentManager.executePendingTransactions();

这一句中的 executePendingTransaction() 是做什么用的呢?
查阅官方文档,给出了对这个方法的解释:

After a FragmentTransaction is committed with FragmentTransaction.commit(), it is scheduled to be executed asynchronously on the process's main thread.

就是这一句,asynchronously:异步。
至于延时多久才能获取到实例,这个就说不准了。

题外话:对于 FragmentTabHost + Fragment 的解决方案,如果 Activity 与 Fragment 没有交互的话,完全可以直接新创建一个 Fragment 实例,而不是费尽心思的去获取到 FragmentTabHost 中已添加的 Fragment 实例。但如果 Fragment 与 Activity 需要频繁交互,建议不要选择这个方案了。

相关文章

网友评论

    本文标题:为什么通过 findFragmentByTag 会获取不到 Fr

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