当我改变设备的方向时,重新加载带有选项卡和片段的活动出现问题。
情况如下:
我有一个活动,在操作栏中有3个选项卡。每个选项卡在主视图中加载不同的片段到FrameLayout
中。如果我不改变设备的方向,一切都正常。但是当我这样做时,Android尝试初始化当前选择的片段两次,从而产生以下错误:
E/AndroidRuntime(2022): Caused by: android.view.InflateException: Binary XML file line #39: Error inflating class fragment
以下是导致错误的步骤序列:
- 我加载了活动,选择了第二个选项卡并改变了设备的方向。
- Android销毁了活动和由第二个选项卡加载的片段实例(从现在开始称为“片段2”)。然后它继续创建活动和片段的新实例。
- 在
Activity.onCreate()
内,我将第一个选项卡添加到操作栏中。当我这样做时,此选项卡会自动选择。这可能会在将来造成问题,但我现在不介意。onTabSelected
被调用,并创建并加载第一个片段的新实例(请参见下面的代码)。 - 我添加所有其他选项卡,而没有触发任何事件,这很好。
- 我调用
ActionBar.selectTab(myTab)
来选择第2个选项卡。 - 首先,对于第一个选项卡,
onTabUnselected()
被调用,然后对于第二个选项卡,onTabSelected()
被调用。此序列将当前片段替换为片段2的实例(请参见下面的代码)。 - 接下来,在片段2实例上调用
Fragment.onCreateView()
,并且片段布局被填充。 - 这里出现问题。Android再次调用片段实例的
onCreate()
,然后调用onCreateView()
,当我尝试(第二次)填充布局时,会产生异常。
我尝试在重新加载活动时不选择第二个选项卡,但第二个片段仍会被初始化并且不会显示(因为我没有选择它的选项卡)。
我找到了这个问题:Android Fragments recreated on orientation change 用户基本上问了我要问的问题,但我不喜欢所选择的答案(只是一个解决方法)。必须有某种方法可以在不使用
android:configChanges
技巧的情况下使其正常工作。如果不清楚,我想知道如何防止重建片段或避免其双重初始化。也很好知道为什么会发生这种情况。 :P
以下是相关代码:
public class MyActivity extends Activity implements ActionBar.TabListener {
private static final String TAG_FRAGMENT_1 = "frag1";
private static final String TAG_FRAGMENT_2 = "frag2";
private static final String TAG_FRAGMENT_3 = "frag3";
Fragment frag1;
Fragment frag2;
Fragment frag3;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// my_layout contains a FragmentLayout inside
setContentView(R.layout.my_layout);
// Get a reference to the fragments created automatically by Android
// when reloading the activity
FragmentManager fm = getFragmentManager();
this.frag1 = fm.findFragmentByTag(MyActivity.TAG_FRAGMENT_1);
this.frag2 = fm.findFragmentByTag(MyActivity.TAG_FRAGMENT_2);
this.frag3 = fm.findFragmentByTag(MyActivity.TAG_FRAGMENT_3)
ActionBar actionBar = getActionBar();
// snip...
// This triggers onTabSelected for the first tab
actionBar.addTab(actionBar.newTab()
.setText("Tab1").setTabListener(this)
.setTag(MyActivity.TAG_FRAGMENT_1));
actionBar.addTab(actionBar.newTab()
.setText("Tab2").setTabListener(this)
.setTag(MyActivity.TAG_FRAGMENT_2));
actionBar.addTab(actionBar.newTab()
.setText("Tab3").setTabListener(this)
.setTag(MyActivity.TAG_FRAGMENT_3));
Tab t = null;
// here I get a reference to the tab that must be selected
// snip...
// This triggers onTabUnselected/onTabSelected
ab.selectTab(t);
}
@Override
protected void onDestroy() {
// Not sure if this is necessary
this.frag1 = null;
this.frag2 = null;
this.frag3 = null;
super.onDestroy();
}
@Override
public void onTabSelected(Tab tab, FragmentTransaction ft) {
Fragment curFrag = getFragmentInstanceForTag(tab.getTag().toString());
if (curFrag == null) {
curFrag = createFragmentInstanceForTag(tab.getTag().toString());
if(curFrag == null) {
// snip...
return;
}
}
ft.replace(R.id.fragment_container, curFrag, tab.getTag().toString());
}
@Override
public void onTabUnselected(Tab tab, FragmentTransaction ft)
{
Fragment curFrag = getFragmentInstanceForTag(tab.getTag().toString());
if (curFrag == null) {
// snip...
return;
}
ft.remove(curFrag);
}
private Fragment getFragmentInstanceForTag(String tag)
{
// Returns this.frag1, this.frag2 or this.frag3
// depending on which tag was passed as parameter
}
private Fragment createFragmentInstanceForTag(String tag)
{
// Returns a new instance of the fragment requested by tag
// and assigns it to this.frag1, this.frag2 or this.frag3
}
}
< p >该片段的代码并不重要,它只是在onCreateView()
方法覆盖上返回一个充气视图。< /p >
onTabSelected
和onTabUnselected
方法。 - StuStirling