许多 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 需要频繁交互,建议不要选择这个方案了。
网友评论