Получение ошибки "Деятельность 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:
- Android.Support.V4.App.FragmentManager
- 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();
}