添加到后台堆栈后如何保持片段状态?


160

我写了一个虚拟活动,可以在两个片段之间切换。当您从FragmentA转到FragmentB时,FragmentA被添加到后台堆栈中。但是,当我返回FragmentA(按回去)时,会创建一个全新的FragmentA,并且它所处的状态会丢失。我感觉自己对这个问题也很满意,但是我提供了完整的代码示例来帮助解决问题:

public class FooActivity extends Activity {
  @Override public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    final FragmentTransaction transaction = getFragmentManager().beginTransaction();
    transaction.replace(android.R.id.content, new FragmentA());
    transaction.commit();
  }

  public void nextFragment() {
    final FragmentTransaction transaction = getFragmentManager().beginTransaction();
    transaction.replace(android.R.id.content, new FragmentB());
    transaction.addToBackStack(null);
    transaction.commit();
  }

  public static class FragmentA extends Fragment {
    @Override public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
      final View main = inflater.inflate(R.layout.main, container, false);
      main.findViewById(R.id.next_fragment_button).setOnClickListener(new View.OnClickListener() {
        public void onClick(View v) {
          ((FooActivity) getActivity()).nextFragment();
        }
      });
      return main;
    }

    @Override public void onSaveInstanceState(Bundle outState) {
      super.onSaveInstanceState(outState);
      // Save some state!
    }
  }

  public static class FragmentB extends Fragment {
    @Override public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
      return inflater.inflate(R.layout.b, container, false);
    }
  }
}

加上一些日志消息:

07-05 14:28:59.722 D/OMG     ( 1260): FooActivity.onCreate
07-05 14:28:59.742 D/OMG     ( 1260): FragmentA.onCreateView
07-05 14:28:59.742 D/OMG     ( 1260): FooActivity.onResume
<Tap Button on FragmentA>
07-05 14:29:12.842 D/OMG     ( 1260): FooActivity.nextFragment
07-05 14:29:12.852 D/OMG     ( 1260): FragmentB.onCreateView
<Tap 'Back'>
07-05 14:29:16.792 D/OMG     ( 1260): FragmentA.onCreateView

它从不调用FragmentA.onSaveInstanceState,并且在您回击时会创建一个新的FragmentA。但是,如果我在FragmentA上并且锁定了屏幕,则会调用FragmentA.onSaveInstanceState。所以很奇怪...我期望添加到后端堆栈的片段不需要重新创建是我错了吗?这是文档所说的:

而如果您在删除片段时确实调用了addToBackStack(),则该片段将停止并在用户向后导航时恢复。


3
@ Jan-Henk那必须要拿的东西呢?例如,的滚动位置ListView。似乎过多的跳跳无法附加滚动侦听器和更新实例变量。
杰克·沃顿

2
@JakeWharton我同意应该会更容易,但是据我所知,这是没有办法的,因为当从后台恢复一个片段时会调用onCreateView。但是我可能是错的:)
Jan-Henk

1
onCreate不会被调用。显然,它正在重用同一实例,但再次调用onCreateView吗?瘸。我想我可以只缓存onCreateView的结果,如果再次调用onCreateView,就可以返回现有视图。
埃里克(Eric)

1
正是我在找几个小时。您可以发布使用实例变量实现此目标的方法吗?
乌玛2013年

1
所以我最近在github.com/frostymarvelous/Folio中开始了自己的实现,遇到了一个问题。在开始出现OOM崩溃之前,我能够创建大约5个复杂的页面/片段。那就是导致我来到这里的原因。隐藏和显示仅仅是不够的。视图太占用内存。
frostymarvelous

Answers:


120

如果从后退堆栈返回一个片段,它不会重新创建该片段,而是重新使用同一实例,并从onCreateView()片段生命周期开始,请参见片段生命周期

因此,如果要存储状态,则应使用实例变量,而不要依赖onSaveInstanceState()


32
该文档的当前版本与该声明矛盾。流程图说明了您所说的内容,但是页面主区域中的文本仅在一次显示Fragment 时才调用onCreateView():developer.android.com/guide/components/fragments.html 我正在努力解决这个问题现在发出问题,并且当从Backstack返回片段时,我看不到任何方法被调用。(Android 4.2)
Colin M.

10
试图记录其行为。显示片段时,始终会调用onCreateView()。
princepiero

4
@ColinM。有什么解决办法吗?
暴雪2014年

9
这对我不起作用。返回片段时,我的实例变量为null!如何保存状态?
Don Rhummy

5
因此,如果我们不中继保存实例,应该如何保存片段状态和数据?
马赫迪

80

与Apple UINavigationController和相比UIViewController,Google在Android软件架构方面做得不好。与Android有关的文档Fragment并没有太大帮助。

当您从FragmentA输入FragmentB时,现有FragmentA实例不会被破坏。当您在FragmentB中按Back返回到FragmentA时,我们不会创建新的FragmentA实例。现有的FragmentA实例onCreateView()将被调用。

关键是我们不应该在FragmentA的视图中再次膨胀视图onCreateView(),因为我们使用的是现有FragmentA的实例。我们需要保存并重用rootView。

以下代码运行良好。它不仅保持片段状态,而且还减少了RAM和CPU负载(因为我们仅在必要时膨胀布局)。我不敢相信Google的示例代码和文档从未提到过,而是总是夸大布局

版本1(请勿使用版本1。请使用版本2)

public class FragmentA extends Fragment {
    View _rootView;
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        if (_rootView == null) {
            // Inflate the layout for this fragment
            _rootView = inflater.inflate(R.layout.fragment_a, container, false);
            // Find and setup subviews
            _listView = (ListView)_rootView.findViewById(R.id.listView);
            ...
        } else {
            // Do not inflate the layout again.
            // The returned View of onCreateView will be added into the fragment.
            // However it is not allowed to be added twice even if the parent is same.
            // So we must remove _rootView from the existing parent view group
            // (it will be added back).
            ((ViewGroup)_rootView.getParent()).removeView(_rootView);
        }
        return _rootView;
    }
}

------ 2005年5月3日更新:-------

正如评论所提到的,有时在中_rootView.getParent()为null onCreateView会导致崩溃。第2版​​按照dell116建议删除了onDestroyView()中的_rootView。在Android 4.0.3、4.4.4、5.1.0上测试。

版本2

public class FragmentA extends Fragment {
    View _rootView;
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        if (_rootView == null) {
            // Inflate the layout for this fragment
            _rootView = inflater.inflate(R.layout.fragment_a, container, false);
            // Find and setup subviews
            _listView = (ListView)_rootView.findViewById(R.id.listView);
            ...
        } else {
            // Do not inflate the layout again.
            // The returned View of onCreateView will be added into the fragment.
            // However it is not allowed to be added twice even if the parent is same.
            // So we must remove _rootView from the existing parent view group
            // in onDestroyView() (it will be added back).
        }
        return _rootView;
    }

    @Override
    public void onDestroyView() {
        if (_rootView.getParent() != null) {
            ((ViewGroup)_rootView.getParent()).removeView(_rootView);
        }
        super.onDestroyView();
    }
}

警告!!!

这是哈克!尽管我在应用程序中使用它,但是您需要仔细测试和阅读评论。


38
对整个片段的rootview进行引用是IMO的一个坏主意。如果您不断向后堆栈中添加多个片段,并且所有片段都保持其rootview(这具有相当大的内存占用),那么您很有可能会遇到OutOfMemoryError错误,因为所有片段都具有rootview引用并且GC无法收集它。我认为更好的方法是始终对视图进行充气(并让Android系统处理视图的创建/销毁),然后onActivityCreated / onViewCreated检查您的数据是否为空。如果是,则加载它,否则将数据设置为视图。
traninho 2014年

15
不要这样!创建片段的视图层次结构时,它包含对活动的内部引用,该活动当时保存了片段。发生配置更改时,通常会重新创建活动。重用旧版式会将僵尸活动及其引用的任何对象保留在内存中。像这样浪费内存会影响性能,并使您的应用成为不在前台时立即终止的最佳选择。
Krylez,2015年

4
@AllDayAmazing这是一个好点。老实说,我现在很困惑。任何人都可以尝试解释为什么对片段的rootview的引用不正确,而仅对rootview的任何子级(无论如何对rootview的引用)都具有引用是可以的吗?
traninho 2015年

2
远离此,除非您想浪费5个小时来找出代码中的错误……然后才发现这是原因。现在,我不得不重构很多东西,因为我使用了这个hack。如果要在查看另一个片段(甚至位于顶部)时保持片段的UI完整,则最好使用fragmentTransaction.add。fragmentTransaction.replace()旨在破坏片段的视图.....不要与系统抗争。
dell116

2
@VinceYuan-我在Android 5.1上使用最新的v7-appcompat库进行了测试,这留下了应该在我的活动的FragmentManager中删除的6个片段实例。即使GC将正确处理(我不相信它将处理),这也会对您的应用程序以及设备造成不必要的内存紧张。只需使用.add()即可完全消除所有这些hacky代码的需要。这样做完全违背了使用FragmentTransaction.replace()首先要做的事情。
dell116

53

我猜想有另一种方法可以实现您想要的。我没有说这是一个完整的解决方案,但就我的情况而言,它可以达到目的。

我所做的是代替了我刚刚添加目标片段的片段。因此,基本上,您将改为使用add()method replace()

我还做了什么。我隐藏了当前片段,并将其添加到堆栈中。

因此,它重叠在当前片段新片段不破坏其观点。(检查其onDestroyView()方法不会被调用。加上它添加到backstate让我恢复片段的优势。

这是代码:

Fragment fragment=new DestinationFragment();
FragmentManager fragmentManager = getFragmentManager();
android.app.FragmentTransaction ft=fragmentManager.beginTransaction();
ft.add(R.id.content_frame, fragment);
ft.hide(SourceFragment.this);
ft.addToBackStack(SourceFragment.class.getName());
ft.commit();

AFAIK系统仅onCreateView()在视图被破坏或未创建时调用。但是这里我们通过不从内存中删除视图来保存视图。因此,它不会创建新视图。

当您从Destination Fragment返回时,它将弹出最后一个FragmentTransaction删除的顶部片段,这将使最上面的(SourceFragment's)视图显示在屏幕上。

评论:正如我所说,这不是一个完整的解决方案,因为它不会删除Source片段的视图,因此比平时占用更多的内存。但还是要达到目的。此外,我们使用完全不同的隐藏视图机制,而不是替换非传统的视图。

因此,这实际上与维护状态无关,而与维护视图有关。


就我而言,通过添加片段而不是替换片段会导致在使用轮询或片段中使用任何其他类型的Web请求时出现问题。添加片段B后,我想在片段A中暂停此轮询。有什么想法吗?
乌玛(Uma)2013年

您如何在FirstFragment中使用轮询?您必须手动执行此操作,因为这两个片段都保留在内存中。因此,您可以使用它们的实例执行必要的操作。这就是线索,可以在main活动中生成一个事件,该事件在添加第二个片段时会有所作为。希望这会有所帮助。
kaushal trivedi 2013年

1
谢谢您的提示=)。我做完了 但这是唯一的方法吗?还有合适的方法吗?另外,当我按下主屏幕按钮并再次启动应用程序时,所有片段都再次激活。通过这种方式在片段B中假设我在这里。Activity A{Fragment A --> Fragment B}当我按下主屏幕按钮后再次启动该应用程序时,两个片段都onResume()被调用,因此它们开始轮询。我该如何控制?
乌玛2013年

1
不幸的是,您不能,系统无法以这种方式在正常行为下工作,它将两个片段都视为活动的直接子代。尽管它起到了维持片段状态的作用,但其他正常情况却很难管理。发现了所有这些问题,现在我的建议是不要这样。抱歉。
kaushal trivedi 2013年

1
当然,最后我会说在找到其他解决方案之前不要采用这种方法。因为很难管理。
kaushal trivedi

7

我建议一个非常简单的解决方案。

使用View参考变量并在OnCreateView中设置视图。检查该变量中是否已存在视图,然后返回相同的视图。

   private View fragmentView;

   public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);

        if (fragmentView != null) {
            return fragmentView;
        }
        View view = inflater.inflate(R.layout.yourfragment, container, false);
        fragmentView = view;
        return view;
    }

1
如果我们不删除onDestroy()中的'fragmentView'变量,就有内存泄漏的可能性
Arun PM

@ArunPM那么如何在onDestroy()中删除fragmentView?if (_rootView.getParent() != null) { ((ViewGroup)_rootView.getParent()).removeView(_rootView); }是否适合清除内存?
MehmetGür

1
@MehmetGür我经常使用此解决方案。到现在我还没有收到任何内存泄漏错误。但是,您可以根据需要使用ArunPM解决方案。我认为他告诉OnDestroy()方法中fragmentView设置为null。
Mandeep Singh

1
当我遵循此方法时,我正在使用LeakCanary检测内存泄漏及其引发的泄漏问题。但随着@Mandeep在评论中提到轻叹我们可以通过分配解决这个问题null,以fragmentView 可变onDestroy()的方法。
阿伦(Arun)

1
据我所知,当一个片段被破坏时,与该片段关联的视图将在中清除onDestroyView()。我们的备份视图变量(此处为fragmentView )并未进行这种清除,当片段回堆/销毁时,它将导致内存泄漏。您可以在LeakCanery简介中的[内存泄漏的常见原因](square.github.io/leakcanary/fundamentals/…)中找到相同的参考。

6

我在包含地图的Fragment中遇到了这个问题,该地图包含太多设置详细信息,无法保存/重新加载。我的解决方案是基本上始终使该片段保持活动状态(类似于@kaushal提到的内容)。

假设您有当前的片段A,并且想要显示片段B。总结一下结果:

  • replace()-删除片段A并将其替换为片段B。一旦再次出现在最前面,就会重新创建片段A
  • add()-(创建并)添加一个片段B,它与片段A重叠,而片段A在后台仍处于活动状态
  • remove()-可用于删除片段B并返回到A。稍后在稍后调用时将重新创建片段B

因此,如果要保留两个“片段”,只需使用hide()/ show()切换它们即可。

优点:保持多个片段运行的简便方法
缺点:您需要使用更多的内存来保持所有片段的运行。可能会遇到问题,例如显示许多大的位图


您能告诉我何时删除片段b并返回到A,然后在片段A中调用哪个方法?当我们删除片段B时,我想采取一些措施
Google

5

onSaveInstanceState() 仅在配置更改时调用。

由于从一个片段更改为另一个片段,因此没有配置更改,因此没有对它的调用onSaveInstanceState()。什么状态没有被保存?你能指定吗?

如果在EditText中输入一些文本,它将自动保存。没有ID的任何UI项都是其视图状态将不被保存的项。


onSaveInstanceState()当系统由于缺少资源而销毁Activity时,也会调用。
Marcel Bro

0

在这里,因为onSaveInstanceState在将片段添加到Backstack时不会调用in片段。还原的开始onCreateView和结束onDestroyView时间while 和onSaveInstanceState之间调用backstack中的片段生命周期。我的解决方案是在中创建实例变量和init 。样例代码:onDestroyViewonDestroyonCreate

private boolean isDataLoading = true;
private ArrayList<String> listData;
public void onCreate(Bundle savedInstanceState){
     super.onCreate(savedInstanceState);
     isDataLoading = false;
     // init list at once when create fragment
     listData = new ArrayList();
}

并检查它onActivityCreated

public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);
    if(isDataLoading){
         fetchData();
    }else{
         //get saved instance variable listData()
    }
}

private void fetchData(){
     // do fetch data into listData
}

0
getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener()
    {
        @Override
        public void onBackStackChanged()
        {
            if (getSupportFragmentManager().getBackStackEntryCount() == 0)
            {
                //setToolbarTitle("Main Activity");
            }
            else
            {
                Log.e("fragment_replace11111", "replace");
            }
        }
    });


YourActivity.java
@Override
public void onBackPressed()
{
 Fragment fragment = getSupportFragmentManager().findFragmentById(R.id.Fragment_content);
  if (fragment instanceof YourFragmentName)
    {
        fragmentReplace(new HomeFragment(),"Home Fragment");
        txt_toolbar_title.setText("Your Fragment");
    }
  else{
     super.onBackPressed();
   }
 }


public void fragmentReplace(Fragment fragment, String fragment_name)
{
    try
    {
        fragmentTransaction = fragmentManager.beginTransaction();
        fragmentTransaction.replace(R.id.Fragment_content, fragment, fragment_name);
        fragmentTransaction.setCustomAnimations(R.anim.enter_from_right, R.anim.exit_to_left, R.anim.enter_from_left, R.anim.exit_to_right);
        fragmentTransaction.addToBackStack(fragment_name);
        fragmentTransaction.commitAllowingStateLoss();
    }
    catch (Exception e)
    {
        e.printStackTrace();
    }
}

0

我的问题是类似的,但是我克服了我的麻烦,却没有保留该片段。假设您有一个包含2个片段的活动-F1和F2。F1最初启动,并说包含一些用户信息,然后在某些条件下,F2弹出提示用户填写其他属性 -他们的电话号码。接下来,您希望该电话号码弹出到F1并完成注册,但是您意识到以前的所有用户信息都丢失了,并且没有他们以前的数据。该片段是从头开始重新创建的,即使您将此信息保存在onSaveInstanceState分发包中,也将返回null onActivityCreated

解决方案: 将所需的信息另存为调用活动中的实例变量。然后将该实例变量传递到您的片段中。

@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);

    Bundle args = getArguments();

    // this will be null the first time F1 is created. 
    // it will be populated once you replace fragment and provide bundle data
    if (args != null) {
        if (args.get("your_info") != null) {
            // do what you want with restored information
        }
    }
}

因此,以我的示例为例:在显示F2之前,我使用回调将用户数据保存在实例变量中。然后我启动F2,用户填写电话号码,然后按保存。我在活动中使用另一个回调,收集此信息并替换我的片段F1,这一次它具有我可以使用的捆绑数据。

@Override
public void onPhoneAdded(String phone) {
        //replace fragment
        F1 f1 = new F1 ();
        Bundle args = new Bundle();
        yourInfo.setPhone(phone);
        args.putSerializable("you_info", yourInfo);
        f1.setArguments(args);

        getFragmentManager().beginTransaction()
                .replace(R.id.fragmentContainer, f1).addToBackStack(null).commit();

    }
}

有关回调的更多信息,请参见:https//developer.android.com/training/basics/fragments/communicating.html


0

首先:仅使用add方法代替FragmentTransaction类的replace方法,然后必须通过addToBackStack方法将secondFragment添加到堆栈中

第二:在后面单击,您必须调用popBackStackImmediate()

Fragment sourceFragment = new SourceFragment ();
final Fragment secondFragment = new SecondFragment();
final FragmentTransaction ft = getChildFragmentManager().beginTransaction();
ft.add(R.id.child_fragment_container, secondFragment );
ft.hide(sourceFragment );
ft.addToBackStack(NewsShow.class.getName());
ft.commit();
                                
((SecondFragment)secondFragment).backFragmentInstanceClick = new SecondFragment.backFragmentNewsResult()
{
        @Override
        public void backFragmentNewsResult()
        {                                    
            getChildFragmentManager().popBackStackImmediate();                                
        }
};

0

使用以下代码替换片段:

Fragment fragment = new AddPaymentFragment();
getSupportFragmentManager().beginTransaction().replace(R.id.frame, fragment, "Tag_AddPayment")
                .addToBackStack("Tag_AddPayment")
                .commit();

活动的onBackPressed()是:

  @Override
public void onBackPressed() {
    android.support.v4.app.FragmentManager fm = getSupportFragmentManager();
    if (fm.getBackStackEntryCount() > 1) {

        fm.popBackStack();
    } else {


        finish();

    }
    Log.e("popping BACKSTRACK===> ",""+fm.getBackStackEntryCount());

}

0
Public void replaceFragment(Fragment mFragment, int id, String tag, boolean addToStack) {
        FragmentTransaction mTransaction = getSupportFragmentManager().beginTransaction();
        mTransaction.replace(id, mFragment);
        hideKeyboard();
        if (addToStack) {
            mTransaction.addToBackStack(tag);
        }
        mTransaction.commitAllowingStateLoss();
    }
replaceFragment(new Splash_Fragment(), R.id.container, null, false);

1
感谢您提供此代码段,它可能会提供一些有限的即时帮助。通过说明为什么这是一个解决问题的好方法,适当的解释将大大提高其长期价值,对于其他存在类似问题的读者来说,这样做将更为有用。请编辑您的答案以添加一些解释,包括您所做的假设。
Machavity

0

完美的解决方案,可在堆栈中找到旧片段并将其加载(如果堆栈中存在)。

/**
     * replace or add fragment to the container
     *
     * @param fragment pass android.support.v4.app.Fragment
     * @param bundle pass your extra bundle if any
     * @param popBackStack if true it will clear back stack
     * @param findInStack if true it will load old fragment if found
     */
    public void replaceFragment(Fragment fragment, @Nullable Bundle bundle, boolean popBackStack, boolean findInStack) {
        FragmentManager fm = getSupportFragmentManager();
        FragmentTransaction ft = fm.beginTransaction();
        String tag = fragment.getClass().getName();
        Fragment parentFragment;
        if (findInStack && fm.findFragmentByTag(tag) != null) {
            parentFragment = fm.findFragmentByTag(tag);
        } else {
            parentFragment = fragment;
        }
        // if user passes the @bundle in not null, then can be added to the fragment
        if (bundle != null)
            parentFragment.setArguments(bundle);
        else parentFragment.setArguments(null);
        // this is for the very first fragment not to be added into the back stack.
        if (popBackStack) {
            fm.popBackStack(null, FragmentManager.POP_BACK_STACK_INCLUSIVE);
        } else {
            ft.addToBackStack(parentFragment.getClass().getName() + "");
        }
        ft.replace(R.id.contenedor_principal, parentFragment, tag);
        ft.commit();
        fm.executePendingTransactions();
    }

像这样使用

Fragment f = new YourFragment();
replaceFragment(f, null, boolean true, true); 
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.