Android

Android FragmentManager BackStackRecord.run throwing NullPointerException

Android FragmentManager BackStackRecord.запустить исключение NullPointerException

Иногда я получаю следующее исключение при работе с фрагментами:

FATAL EXCEPTION: main
java.lang.NullPointerException
at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:591)
at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1416)
at android.support.v4.app.FragmentManagerImpl$1.run(FragmentManager.java:420)
at android.os.Handler.handleCallback(Handler.java:615)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:4745)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
at dalvik.system.NativeStart.main(Native Method)

Исключение возникает, когда run() of BackStackRecord вызывается через execPendingTransactions(), когда он пытается удалить фрагмент из менеджера.

case OP_REMOVE: {
Fragment f = op.fragment;
f.mNextAnim = op.exitAnim; <----
mManager.removeFragment(f, mTransition, mTransitionStyle);
}
break;

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

Переведено автоматически
Ответ 1

Отвечаю на мой собственный вопрос:

Это исключение (в конечном итоге) выдается при вызове FragmentTransaction.remove(null); и FragmentTransaction.commit();

РЕДАКТИРОВАТЬ: А также, как дважды обведено и на что указывает shinyuX в комментарии; при вызове методов show(null) or add(null), attach(null) и detach(null), и, вероятно, также hide(null)

После вызова commit() транзакция будет помещена в очередь во FragmentManager. В результате, когда операция обрабатывается после вашего явного вызова FragmentManager.executePendingTransactions() или когда поток очереди FragmentManager вызывает ее, она выдает NullPointerException.

В моем случае я поддерживал состояния фрагмента в глобальном объекте. Там я проверил, отображается фрагмент или нет, а затем удалил видимые фрагменты. Но поскольку я запустил новую FragmentActivity, для этих состояний все еще было установлено значение true, пока они не были видны. Так что это ошибка разработки.

Помимо исправления ошибки проектирования, решение было простым: проверьте, FragmentManager.findFragmentByTag() возвращен ли null перед удалением фрагмента.

Ответ 2

Единственная причина, по которой это происходит, заключается в вызове

getSupportFragmentManager().beginTransaction().remove(fragment)

пока fragment равно null

Ответ 3

Я не использую tag для создания фрагментов (они работают как контейнеры панели вкладок).

Итак, это работает при смене вкладки, но если я нажимаю кнопку "Назад", я получаю ту же ошибку.

В методе onDestroyView я нашел экземпляр фрагмента с помощью FragmentManager#findFragmentById , однако FragmentManager#findFragmentByTag, конечно, возвращает null.

class MyFragment extends ListFragment {

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

if (this.mapFragment != null
&& getFragmentManager().findFragmentById(
this.mapFragment.getId()) != null) {

getFragmentManager().beginTransaction().remove(this.mapFragment)
.commit();
this.mapFragment = null;
}

}
}
2023-10-09 19:05 java android android-fragments