Получение ошибки "Деятельность Java.lang.IllegalStateException была уничтожена" при использовании вкладок с ViewPager

У меня есть приложение, которое состоит из использования ActionBarSherlock в режиме вкладок. У меня есть 5 вкладок, и содержимое каждой вкладки обрабатывается с использованием фрагментов. Для tab2 у меня есть фрагмент, xml-файл которого содержит элемент ViewPager, который, в свою очередь, содержит несколько фрагментов страниц. Когда я первоначально запускаю приложение, я могу переключаться между вкладками без проблем, но когда я нажимаю на tab2 во второй раз, я получаю ошибку, упомянутую выше. Основным видом деятельности является следующее:

public class MainActivity extends SherlockFragmentActivity
{
    @Override
    protected void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        ActionBar actionBar = getSupportActionBar();

        ActionBar.Tab tab1 = actionBar.newTab().setText("Tab1");
        ActionBar.Tab tab3 = actionBar.newTab().setText("Tab3");
        ActionBar.Tab tab2 = actionBar.newTab().setText("Tab2");
        ActionBar.Tab tab4 = actionBar.newTab().setText("Tab4");
        ActionBar.Tab tab5 = actionBar.newTab().setText("Tab5");

        Fragment fragment1 = new Tab1();
        Fragment fragment3 = new Tab3();
        Fragment fragment2 = new Tab2();
        Fragment fragment5 = new Tab5();
        Fragment fragment4 = new Tab4();

        tab1.setTabListener(new MyTabListener(fragment1));
        tab3.setTabListener(new MyTabListener(fragment3));
        tab2.setTabListener(new MyTabListener(fragment2));
        tab5.setTabListener(new MyTabListener(fragment5));
        tab4.setTabListener(new MyTabListener(fragment4));

        actionBar.addTab(tab1);
        actionBar.addTab(tab2);
        actionBar.addTab(tab3);
        actionBar.addTab(tab4);
        actionBar.addTab(tab5); 

        actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
    }

    class MyTabListener implements ActionBar.TabListener
    {
        Fragment fragment;

        public MyTabListener(Fragment fragment)
        {
            this.fragment = fragment;
        }

        @Override
        public void onTabSelected(com.actionbarsherlock.app.ActionBar.Tab tab,FragmentTransaction ft) 
        {
            ft.replace(R.id.fragment_container,fragment);
        }

        @Override
        public void onTabUnselected(com.actionbarsherlock.app.ActionBar.Tab tab,FragmentTransaction ft) 
        {

        }

        @Override
        public void onTabReselected(com.actionbarsherlock.app.ActionBar.Tab tab,FragmentTransaction ft) 
        {

        }
    }
}

Класс фрагмента без ViewPager выглядит следующим образом:

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

Класс фрагмента с ViewPager выглядит следующим образом:

public class Tab2 extends Fragment 
{
    ViewPager mViewPager;
    private MyFragmentPagerAdapter mMyFragmentPagerAdapter;  
    private static int NUMBER_OF_PAGES = 5;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState)
    {
        View view =  inflater.inflate(R.layout.activity_tab2, container, false); 
        return view;
    }

    @Override
    public void onViewCreated(View view,Bundle savedInstanceState)
    {
        super.onViewCreated(view, savedInstanceState);
        mViewPager = (ViewPager) view.findViewById(R.id.viewpager);
        mMyFragmentPagerAdapter = new MyFragmentPagerAdapter(getChildFragmentManager());  
        mViewPager.setAdapter(mMyFragmentPagerAdapter);  
    }

    private static class MyFragmentPagerAdapter extends FragmentPagerAdapter 
    {    
        public MyFragmentPagerAdapter(FragmentManager fm) 
        {  
             super(fm);  
        }  

        @Override  
        public Fragment getItem(int index) 
        {  
             return PageFragment.newInstance("My Message " + index);
        }  

        @Override  
        public int getCount() 
        {  
             return NUMBER_OF_PAGES;  
        }  
   }
}

Из того, что я прочитал в разных местах (и, пожалуйста, исправьте меня, если я ошибаюсь), это происходит потому, что менеджер фрагментов на втором проходе пытается повторно использовать фрагменты из действия, которого больше не существует, что приводит к ошибке. Но я не уверен, почему это происходит здесь, так как я не использую фрагментную активность. Согласно logcat, ошибка находится в классе Tab2, метод onViewCreated в строке, которая говорит mViewPager.setAdapter(mMyFragmentPagerAdapter). Любая помощь с благодарностью... Спасибо.

03-04 12:01:05.468: E/AndroidRuntime(2474): FATAL EXCEPTION: main
03-04 12:01:05.468: E/AndroidRuntime(2474): java.lang.IllegalStateException: Activity has been destroyed
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentManagerImpl.enqueueAction(FragmentManager.java:1342)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.BackStackRecord.commitInternal(BackStackRecord.java:595)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.BackStackRecord.commitAllowingStateLoss(BackStackRecord.java:578)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentPagerAdapter.finishUpdate(FragmentPagerAdapter.java:139)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.view.ViewPager.populate(ViewPager.java:1011)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.view.ViewPager.populate(ViewPager.java:880)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.view.ViewPager.setAdapter(ViewPager.java:433)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at com.example.tabs.Tab2.onViewCreated(Tab2.java:31)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:925)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1088)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:682)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1444)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.support.v4.app.FragmentManagerImpl$1.run(FragmentManager.java:429)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.os.Handler.handleCallback(Handler.java:587)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.os.Handler.dispatchMessage(Handler.java:92)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.os.Looper.loop(Looper.java:123)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at android.app.ActivityThread.main(ActivityThread.java:3687)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at java.lang.reflect.Method.invokeNative(Native Method)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at java.lang.reflect.Method.invoke(Method.java:507)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:842)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
03-04 12:01:05.468: E/AndroidRuntime(2474):     at dalvik.system.NativeStart.main(Native Method)

14 ответов

Решение

Это, похоже, ошибка во вновь добавленной поддержке вложенных фрагментов. В основном ребенок FragmentManager заканчивается нарушенным внутренним состоянием, когда он оторван от деятельности. Краткосрочный обходной путь, который исправил это для меня, должен добавить следующее к onDetach() каждого Fragment который ты называешь getChildFragmentManager() на:

@Override
public void onDetach() {
    super.onDetach();

    try {
        Field childFragmentManager = Fragment.class.getDeclaredField("mChildFragmentManager");
        childFragmentManager.setAccessible(true);
        childFragmentManager.set(this, null);

    } catch (NoSuchFieldException e) {
        throw new RuntimeException(e);
    } catch (IllegalAccessException e) {
        throw new RuntimeException(e);
    }
}

У меня точно такая же проблема. Единственный обходной путь, который я нашел, - это заменить фрагменты новым экземпляром при каждом изменении вкладок.

ft.replace(R.id.fragment_container, Fragment.instantiate(PlayerMainActivity.this, fragment.getClass().getName()));

Не настоящее решение, но я не нашел способ повторно использовать предыдущий экземпляр фрагмента...

Я столкнулся с той же проблемой при звонке super.onCreate() в конце моего метода. Причина: attachActivity() вызывается в onCreate() FragmentActivity. При переопределении onCreate() и, например, создавая вкладки, менеджер вкладок будет пытаться переключиться на фрагмент, не привязывая активность к FragmentManager.

Простое решение: переместить звонок на super.onCreate() руководителю функционального органа.

В общем, кажется, есть множество причин, по которым эта проблема может возникнуть. Это просто еще один...

Матиас

Хотел добавить, что моя проблема была в деятельности, где я пытался сделать FragmentTransaction в onCreate, прежде чем я позвонил super.onCreate(), Я только что переехала super.onCreate() к началу работы и работал нормально.

Я столкнулся с той же проблемой, и позже узнал, что я пропустил звонок super.onCreate( savedInstanceState ); в onCreate() FragmentActivity.

У меня была эта ошибка, потому что я использовал LocalBroadcastManager, и я сделал:

unregisterReceiver(intentReloadFragmentReceiver);

вместо:

LocalBroadcastManager.getInstance(this).unregisterReceiver(intentReloadFragmentReceiver);

Я знаю, что это старый пост, но предложенные ответы не сработали с моей стороны. Я хочу оставить это здесь на всякий случай, если кто-то сочтет это полезным.

Что я сделал, это:

@Override
public void onResume() {
    super.onResume();
    // add all fragments
    FragmentTransaction fragmentTransaction = getChildFragmentManager().beginTransaction();
    for(Fragment fragment : fragmentPages){
        String tag = fragment.getClass().getSimpleName();
        fragmentTransaction.add(R.id.contentPanel, fragment, tag);
        if(fragmentPages.indexOf(fragment) != currentPosition){
            fragmentTransaction.hide(fragment);
        } else {
            lastTag = tag;
        }
    }
    fragmentTransaction.commit();
}

Затем в:

@Override
public void onPause() {
    super.onPause();
    // remove all attached fragments
    for(Fragment fragment: fragmentPages){
        getChildFragmentManager().beginTransaction().remove(fragment).commit();
    }
}

Я заставляю фрагмент, содержащий дочерний фрагмент, в NULL в onPause, и это решает мою проблему

fragment = null;

Я получил ту же ошибку при попытке доступа к ребенку FragmentManager до того, как фрагмент был полностью инициализирован (т.е. прикреплен к Activity или, по крайней мере, onCreateView() называется). Остальное FragmentManager инициализируется с null Деятельность, вызывающая вышеупомянутое исключение.

Этот сводил меня с ума из-за Ксамарина.

Я столкнулся с этим с помощью реализации ViewPager для TabLayout WITHIN Fragment, которая сама реализована в DrawerLayout:

 - DrawerLayout
   - DrawerFragment
     - TabLayout
     - TabViewPager
       - TabViewPagerFragments

Поэтому вы должны реализовать следующий код в своем DrawerFragment. Будьте внимательны, чтобы выбрать правильный FragmentManager-Path. Потому что у вас может быть две разные ссылки FragmentManager:

  1. Android.Support.V4.App.FragmentManager
  2. Android.App.FragmentManager

-> Выберите тот, который вы используете. Если вы хотите использовать ChildFragmentManager, вы должны использовать объявление класса Android.App.FragmentManager для вашего ViewPager!

Android.Support.V4.App.FragmentManager

Реализуйте следующий метод в своем " основном " фрагменте - в этом примере: DrawerFragment

public override void OnDetach() {
    base.OnDetach();
    try {
        Fragment x = this;
        var classRefProp = typeof(Fragment).GetProperty("class_ref", BindingFlags.NonPublic | BindingFlags.Static);
        IntPtr classRef = (IntPtr)classRefProp.GetValue(x);
        var field = JNIEnv.GetFieldID(classRef, "mChildFragmentManager", "Landroid/support/v4/app/FragmentManagerImpl;");
        JNIEnv.SetField(base.Handle, field, IntPtr.Zero);
    }
    catch (Exception e) {
        Log.Debug("Error", e+"");
    }
}

Android.App.FragmentManager

public class TabViewPager: Android.Support.V13.App.FragmentPagerAdapter {}

Это означает, что вам нужно было запустить ViewPager с Android.App.FragmentManager.

Реализуйте следующий метод в своем " основном " фрагменте - в этом примере: DrawerFragment

public override void OnDetach() {
    base.OnDetach();
    try {
        Fragment x = this;
        var classRefProp = typeof(Fragment).GetProperty("class_ref", BindingFlags.NonPublic | BindingFlags.Static);
        IntPtr classRef = (IntPtr)classRefProp.GetValue(x);
        var field = JNIEnv.GetFieldID(classRef, "mChildFragmentManager", "Landroid/app/FragmentManagerImpl;");
        JNIEnv.SetField(base.Handle, field, IntPtr.Zero);
    }
    catch (Exception e) {
        Log.Debug("Error", e+"");
    }
}

Ошибка исправлена ​​в последней версии androidx. И знаменитый обходной путь теперь вызовет сбой. так что нам это не нужно сейчас.

я обнаружил, что у меня был таймер, работающий в фоновом режиме. когда активность убита, но таймер все еще работает. в обратном вызове завершения таймера я обращаюсь к объекту фрагмента, чтобы выполнить некоторую работу, и вот ошибка !!!! фрагмент существует, но активности нет.

если у вас есть служба таймера или какие-либо фоновые потоки, убедитесь, что у вас нет доступа к объектам фрагментов.

У меня была эта проблема и понял, что это потому, что я звонил setContentView(int id) дважды в моем Activity"s onCreate

У меня была эта проблема, и я не мог найти решение здесь, поэтому я хочу поделиться своим решением на случай, если у кого-то еще возникнет эта проблема снова.

У меня был этот код:

public void finishAction() {
  onDestroy();
  finish();
}

и решил проблему, удалив строку "onDestroy();"

public void finishAction() {
  finish();
}

Причина, по которой я написал исходный код: я знаю, что когда вы выполняете "finish()", действие вызывает "onDestroy()", но я использую потоки, и я хотел убедиться, что все потоки уничтожены, прежде чем начинать следующее действие и похоже, что "finish()" не всегда немедленный. Мне нужно обработать / уменьшить количество "растровых изображений" и отображать большие "растровые изображения", и я работаю над улучшением использования памяти в моем приложении

Теперь я уничтожу потоки, используя другой метод, и я выполню этот метод из "onDestroy();" и когда я думаю, что мне нужно убить все темы.

public void finishAction() {
  onDestroyThreads();
  finish();
}