У меня проблема с передачей данных во фрагменты.Он падает на 0,1% всех времен производства.Допустим, при открытии 100к активности это происходит 100 раз.Это выглядит не очень часто, но это очень беспокоит меня, и я думаю, что я делаю что-то не так с инициализацией фрагментов с данными.Дело в том, что я создаю фрагменты только один раз, а все остальное время мне нужно передать им данные, я делаю это следующим образом: myFragmentInstance.setData(Object someData);
И происходит сбой, потому что он говорит, что эти элементы представления во фрагменте не найдены, и ониNULL, но все должно быть хорошо, если я не воссоздал их.Я не вращаю свой телефон или недостаточно памяти на нем.Это происходит при повторном подключении к сети, потому что при повторном подключении к сети я собираюсь на сервер для получения свежих данных, а затем устанавливаю эти новые данные для своих фрагментов.У меня есть фотографии полей двух фрагментов, которые я использую, возможно, некоторые из вас знают, что эти данные могут сказать о состоянии фрагментов в момент сбоя.Я использую библиотеку ButterKnife
для инициализации полей фрагментов и действий, а не для инициализации findById
, возможно, это имеет какое-то влияние или нет?
Вот ссылка на простой проект (только эта проблемана github): https://github.com/yozhik/Reviews/tree/master/app/src/main/java/com/ylet/sr/review
Описание:
CollapsingActivity
- активность с Collapsing AppBarLayout
.Который загружает один или два фрагмента в «fragment_content_holder
» и имеет TabLayout
для переключения между фрагментами в пейджере представления.
В методе действия onCreate()
- Я просто имитирую запрос к серверу (loadData
), и когда загружаются некоторые фальшивые данные - я показываю фрагменты в пейджере представления, при первом вызове - я создаю новые TabMenuAdapter
расширения FragmentPagerAdapter
, заполняю их фрагментами и сохраняю ссылки на экземпляры.При следующем вызове - я не создаю фрагменты с нуля, а просто заполняю их свежими данными.
MenuFragment1, MenuFragment1
- два фрагмента.MenuFragment1
- имеет метод public void setupData(SomeCustomData data)
, для установки новых данных, без воссоздания фрагмента при повторном подключении к сети.
NetworkStateReceiver
- прослушивание изменений сети и отправка уведомлений.
TabMenuAdapter
-просто простой класс для хранения фрагментов.
05-11 18:11:05.088 12279-12279/com.myProjectName E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.myProjectName, PID: 12279
java.lang.IllegalStateException: Fatal Exception thrown on Scheduler.
at io.reactivex.android.schedulers.HandlerScheduler$ScheduledRunnable.run(HandlerScheduler.java:111)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5268)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:902)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:697)
Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.widget.TextView.setText(java.lang.CharSequence)' on a null object reference
at com.yozhik.myProjectName.view.fragments.MyFinalTermsFragment.setupMyInformation(MyFinalTermsFragment.java:145)
at com.yozhik.myProjectName.view.fragments.MyFinalTermsFragment.setupWithData(MyFinalTermsFragment.java:133)
at com.yozhik.myProjectName.view.activity.MyFinalActivity.onDataLoaded(MyFinalActivity.java:742)
at com.yozhik.myProjectName.presenter.MyFinalPresenter$1.onNext(MyFinalPresenter.java:55)
at com.yozhik.myProjectName.presenter.MyFinalPresenter$1.onNext(MyFinalPresenter.java:47)
at io.reactivex.internal.operators.observable.ObservableObserveOn$ObserveOnObserver.drainNormal(ObservableObserveOn.java:200)
at io.reactivex.internal.operators.observable.ObservableObserveOn$ObserveOnObserver.run(ObservableObserveOn.java:252)
at io.reactivex.android.schedulers.HandlerScheduler$ScheduledRunnable.run(HandlerScheduler.java:109)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5268)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:902)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:697)
05-11 18:11:07.953 2155-3877/? E/WifiStateMachine: Did not find remoteAddress {192.168.200.1} in /proc/net/arp
05-11 18:11:07.966 2155-3877/? E/WifiStateMachine: WifiStateMachine CMD_START_SCAN source -2 txSuccessRate=3800.62 rxSuccessRate=4732.06 targetRoamBSSID=any RSSI=-68
05-11 18:11:07.967 2155-3877/? E/WifiStateMachine: WifiStateMachine L2Connected CMD_START_SCAN source -2 2324, 2325 -> obsolete
05-11 18:11:08.021 2155-3896/? E/ConnectivityService: Unexpected mtu value: 0, wlan0
05-11 18:11:08.579 13514-13366/? E/WakeLock: release without a matched acquire!
Фрагмент, вызывающий сбой в методе setupData , поскольку иногда data_1_txt имеет значение NULL.
public class MenuFragment1 extends Fragment {
public SomeCustomData transferedDataFromActivity;
private TextView data_1_txt;
public static MenuFragment1 newInstance(SomeCustomData data) {
Log.d("TEST", "MenuFragment1.newInstance");
MenuFragment1 fragment = new MenuFragment1();
Bundle args = new Bundle();
args.putSerializable("DATA_FROM_ACTIVITY", data);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d("TEST", "MenuFragment1.onCreate");
if (getArguments() != null) {
this.transferedDataFromActivity = (SomeCustomData) getArguments().getSerializable("DATA_FROM_ACTIVITY");
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
Log.d("TEST", "MenuFragment1.onCreateView");
View v = inflater.inflate(R.layout.menu_fragment_1, container, false);
data_1_txt = (TextView) v.findViewById(R.id.data_1_txt);
setupInOnCreateView();
return v;
}
protected void setupInOnCreateView() {
Log.d("TEST", "MenuFragment1.setupInOnCreateView");
//initialization of all view elements of layout with data is happens here.
setupData(transferedDataFromActivity);
}
public void setupData(SomeCustomData data) {
Log.d("TEST", "MenuFragment1.setupData");
this.transferedDataFromActivity = data;
if (transferedDataFromActivity != null) {
data_1_txt.setText(transferedDataFromActivity.Name);
}
}
}
Макет фрагмента:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/green"
android:orientation="vertical">
<TextView
android:id="@+id/data_1_txt"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/yellow"
android:text="Test"
android:textSize="20sp" />
<include layout="@layout/description_layout" />
</LinearLayout>