【问题标题】:Why does rotation cause Android fragment replacement to fail?为什么旋转会导致Android Fragment替换失败?
【发布时间】:2015-06-06 01:40:38
【问题描述】:

我已经编写了一个简单的程序,它使用单个活动和两个片段进行片段替换。一个片段有一个按钮,当按下该按钮时,将用第二个片段替换该片段。

如果应用程序启动并单击按钮,这将按预期工作。

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

这里发生了什么?是否存在我未能解决的片段生命周期问题?

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;
    }
}

ReplacementFragment.java

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>

【问题讨论】:

  • 使用 onSaveInstanceState?
  • @KristyWelsh 有什么建议吗?我认为没有任何状态需要保存。
  • 显示哪个片段的状态?

标签: java android android-fragments


【解决方案1】:

答案很简单。 UI 状态在屏幕旋转时保持不变,这意味着如果您什么都不做,旋转后屏幕仍会显示第二个片段。由于您在 Activity onCreate 方法中添加了第一个片段,它将被添加到已经显示的内容中,因此两个片段重叠。

你可以这样做:

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();             

}

【讨论】:

  • 这向我解释了为什么在替换片段后屏幕旋转时两个片段都会显示。但是,我不明白为什么在旋转后调用替换失败。根据文档,我认为 replace 应该删除所有片段并用参数替换它们。 “替换已添加到容器中的现有片段。这与为所有当前添加的片段调用 remove(Fragment) 相同,这些片段使用相同的 containerViewId 添加,然后使用给定的相同参数添加(int,Fragment,String)在这里。”
  • 同样的原因。当您旋转另一个第一个片段时,会添加另一个片段,但您看不到,因为它们看起来相同。当你替换时,只有一个被第二个片段替换,然后你有相同的重叠。
  • 所以您是说replace 的文档有误?它不会删除所有当前添加的片段吗?这与我看到的行为一致。
  • 不,我不是这个意思。我是说您正在替换,但由于在屏幕旋转后添加了两个 MainActivityFragment,它将只替换其中一个
  • 您是说它将取代一个(我认为这就是发生的事情)。我不明白这与文档有何一致,文档中说“这与为所有当前添加的使用相同 containerViewId 添加的片段调用 remove(Fragment) 基本相同......”。我的理解是所有 MainActivityFragments 都应该被删除,而不仅仅是一个。
【解决方案2】:

@Emanuel 正确的是 UI 状态在屏幕旋转时保持不变。然而,发生的只是你的 Activity 的 onCreate 总是在轮换时被调用,它总是添加你的 startingFragment

解决方案很简单。只需在 onCreate 中添加一个条件,以便在它作为活动生命周期的一部分被调用时处理这种情况:

if (savedInstanceState == null)
        {
            FragmentTransaction t = getFragmentManager().beginTransaction();
            t.add(R.id.fragment_container, startingFragment, "start").commit();
        }

你可能想考虑做

t.replace(R.id.fragment_container, startingFragment, "start").commit();

虽然不是

t.add(R.id.fragment_container, startingFragment, "start").commit();

【讨论】:

  • 出于好奇,为什么同一片段的多次添加不会导致“java.lang.IllegalStateException: Fragment already added”?
  • 虽然 savedInstanceState == null 也可以工作,但它不如检查 ui 树中是否存在片段和 java.lang.IllegalStateException: Fragment already added is not throwed because you add a new fragment不是同一个。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-11-26
  • 2018-03-30
  • 1970-01-01
  • 2012-12-03
相关资源
最近更新 更多