Java 为什么旋转会导致Android碎片替换失败?
我已经把一个简单的程序组合在一起,它使用一个活动和两个片段替换片段。一个片段有一个按钮,按下该按钮时,将用第二个片段替换该片段 如果启动应用程序并单击按钮,则此操作将按预期进行。 如果应用程序启动,设备旋转(横向或继续纵向),然后单击按钮,则碎片替换失败,同时显示两个碎片。 这是怎么回事?是否存在我未能解决的碎片生命周期问题 MainActivity.javaJava 为什么旋转会导致Android碎片替换失败?,java,android,android-fragments,Java,Android,Android Fragments,我已经把一个简单的程序组合在一起,它使用一个活动和两个片段替换片段。一个片段有一个按钮,按下该按钮时,将用第二个片段替换该片段 如果启动应用程序并单击按钮,则此操作将按预期进行。 如果应用程序启动,设备旋转(横向或继续纵向),然后单击按钮,则碎片替换失败,同时显示两个碎片。 这是怎么回事?是否存在我未能解决的碎片生命周期问题 MainActivity.java package edu.mindlab.fragmenttest; import android.app.FragmentTran
package edu.mindlab.fragmenttest;
import android.app.FragmentTransaction;
import android.support.v7.app.ActionBarActivity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
public class MainActivity extends ActionBarActivity{
private MainActivityFragment startingFragment;
private ReplacementFragment replacementFragment;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
startingFragment = new MainActivityFragment();
replacementFragment = new ReplacementFragment();
FragmentTransaction t = getFragmentManager().beginTransaction();
t.add(R.id.fragment_container, startingFragment, "start").commit();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
//noinspection SimplifiableIfStatement
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
public void replaceFragment(View view){
if(!replacementFragment.isAdded()) {
FragmentTransaction t = getFragmentManager().beginTransaction();
t.replace(R.id.fragment_container, replacementFragment, "replacementTest").commit();
}
}
}
MainActivityFragment.java
package edu.mindlab.fragmenttest;
import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
public class MainActivityFragment extends Fragment {
public MainActivityFragment() {
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_main, container, false);
return view;
}
}
package edu.mindlab.fragmenttest;
import android.app.Activity;
import android.net.Uri;
import android.os.Bundle;
import android.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
public class ReplacementFragment extends Fragment {
public static ReplacementFragment newInstance() {
ReplacementFragment fragment = new ReplacementFragment();
return fragment;
}
public ReplacementFragment() {
// Required empty public constructor
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_replacement, container, false);
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
}
@Override
public void onDetach() {
super.onDetach();
}
}
ReplacementFragment.java
package edu.mindlab.fragmenttest;
import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
public class MainActivityFragment extends Fragment {
public MainActivityFragment() {
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_main, container, false);
return view;
}
}
package edu.mindlab.fragmenttest;
import android.app.Activity;
import android.net.Uri;
import android.os.Bundle;
import android.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
public class ReplacementFragment extends Fragment {
public static ReplacementFragment newInstance() {
ReplacementFragment fragment = new ReplacementFragment();
return fragment;
}
public ReplacementFragment() {
// Required empty public constructor
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_replacement, container, false);
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
}
@Override
public void onDetach() {
super.onDetach();
}
}
activity_main.xml
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/fragment_container">
</FrameLayout>
fragment_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingBottom="@dimen/activity_vertical_margin"
tools:context=".MainActivityFragment">
<TextView
android:text="@string/start_string"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/textView"/>
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/replacement_fragment"
android:id="@+id/button"
android:layout_below="@+id/textView"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true"
android:layout_marginTop="56dp"
android:onClick="replaceFragment"/>
</RelativeLayout>
fragment_replacement.xml
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="edu.mindlab.fragmenttest.ReplacementFragment">
<TextView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:text="@string/replacement_fragment"/>
</FrameLayout>
答案很简单。UI状态在屏幕旋转时保持不变,这意味着如果您什么也不做,屏幕在旋转后仍然会显示第二个片段。由于您在ActivityOnCreate方法中添加了第一个片段,它将被添加到已经显示的内容中,因此两个片段重叠 你可以这样做:
FragmentManager mgr = getFragmentManager();
if (mgr.findFragmentByTag("start") == null &&
mgr.findFragmentByTag("replacementTest") == null) {
FragmentTransaction t = mgr.beginTransaction();
t.add(R.id.fragment_container, startingFragment, "start").commit();
}
@Emanuel认为UI状态在屏幕旋转时保持不变是正确的。然而,发生的事情很简单,您的活动的onCreate总是在轮换时被调用,这总是添加您的
startingFragment
解决办法很简单。只需在onCreate中添加一个条件,以便在将其作为活动生命周期的一部分调用时处理该情况:
if (savedInstanceState == null)
{
FragmentTransaction t = getFragmentManager().beginTransaction();
t.add(R.id.fragment_container, startingFragment, "start").commit();
}
<>你可能想考虑做< /P>
t.replace(R.id.fragment_container, startingFragment, "start").commit();
虽然不是
t.add(R.id.fragment_container, startingFragment, "start").commit();
使用onSaveInstanceState?@KristyWelsh有什么建议吗?我认为没有任何状态需要保存。显示哪个片段的状态?这向我解释了为什么在替换片段后,如果屏幕旋转,两个片段都会显示。然而,我不明白为什么在旋转后调用replace时失败。根据文档,我认为replace应该删除所有片段,并用参数替换它们。“替换添加到容器中的现有片段。这基本上与为所有当前添加的片段调用remove(fragment)相同,这些片段是使用相同的ContainerWebID添加的,然后使用此处给出的相同参数添加(int、fragment、String)。”同样的原因。旋转时,会添加另一个第一个片段,但您看不到,因为它们看起来完全相同。当你替换时,只有一个片段被第二个片段替换,然后你有相同的重叠。那么你是说替换的文档是错误的?它不会删除所有当前添加的片段吗?这与我看到的行为一致。不,我不是这么说的。我是说你正在替换,但是由于在屏幕旋转后添加了两个主要的ActivityFragment,它将只替换其中一个,你是说它将替换一个(我认为这就是发生的情况)。我不明白这是如何与文档一致的,文档中说“这基本上与为所有当前添加的片段调用remove(Fragment)相同,这些片段是使用相同的containerwid添加的……”。我的理解是应该删除所有的mainActivityFragment,而不仅仅是一个。出于好奇,为什么对同一个片段进行多次添加不会导致“java.lang.IllegalStateException:已添加的片段”?虽然savedInstanceState==null也可以工作,但它不如检查ui树和java.lang.IllegalStateException中是否存在片段那么健壮:不会抛出已添加的片段,因为您添加了一个不同的新片段。