Java 为什么旋转会导致Android碎片替换失败?

Java 为什么旋转会导致Android碎片替换失败?,java,android,android-fragments,Java,Android,Android Fragments,我已经把一个简单的程序组合在一起,它使用一个活动和两个片段替换片段。一个片段有一个按钮,按下该按钮时,将用第二个片段替换该片段 如果启动应用程序并单击按钮,则此操作将按预期进行。 如果应用程序启动,设备旋转(横向或继续纵向),然后单击按钮,则碎片替换失败,同时显示两个碎片。 这是怎么回事?是否存在我未能解决的碎片生命周期问题 MainActivity.java package edu.mindlab.fragmenttest; import android.app.FragmentTran

我已经把一个简单的程序组合在一起,它使用一个活动和两个片段替换片段。一个片段有一个按钮,按下该按钮时,将用第二个片段替换该片段

如果启动应用程序并单击按钮,则此操作将按预期进行。

如果应用程序启动,设备旋转(横向或继续纵向),然后单击按钮,则碎片替换失败,同时显示两个碎片。

这是怎么回事?是否存在我未能解决的碎片生命周期问题

MainActivity.java

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中是否存在片段那么健壮:不会抛出已添加的片段,因为您添加了一个不同的新片段。