Поля расположения фрагмента NULL при инициализации - PullRequest
0 голосов
/ 12 мая 2018

У меня проблема с передачей данных во фрагменты.Он падает на 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!

enter image description here enter image description here enter image description here enter image description here

Фрагмент, вызывающий сбой в методе 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>

Ответы [ 3 ]

0 голосов
/ 14 мая 2018

Я почти уверен, что ваши проблемы связаны с хранением ссылок на фрагменты в массиве. Фрагменты имеют жизненные циклы, и ссылки не сохраняются. Как вы сказали, сложно воспроизвести и отследить, что именно происходит, но, возможно, вам это не нужно. Несколько советов о том, как это исправить:

  1. Не хранить ссылки на фрагменты. Практически следуйте примеру на странице Google (https://developer.android.com/training/animation/screen-slide) и создайте новый фрагмент каждый раз, когда он запрашивается.

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

  3. Если вам нужно получить доступ к фрагментам страницы из основного фрагмента (или действия) вместо хранения ссылок, используйте `findFragmentByTag ', который всегда будет возвращать активный в данный момент экземпляр фрагмента.

0 голосов
/ 20 мая 2018

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

Рассмотрим этот сценарий.

  • Вы делаете сетевой запрос, который вносит некоторые изменения в представление в конце.
  • Вы переключаете пейджер. (Фрагмент отделен, просмотры разрушен) * * 1 010
  • Здесь идет обратный вызов из сетевого запроса. Угадай, что! Авария .

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

//in the base class
protected boolean isSafe()
    {
        return !(this.isRemoving() || this.getActivity() == null || this.isDetached()
                || !this.isAdded() || this.getView() == null);
    }

//usage in derived classes
onNewtworkResult(Result result) {
    if(!isSafe())
        return;
    //rest of the code
}

Кроме того, вы также можете заключить потенциальный код в попытку. Но это больше похоже на слепой выстрел (по крайней мере, в этом случае).

0 голосов
/ 13 мая 2018

По моему опыту, если во фрагментах есть ошибка, это обычно происходит из-за предварительной загрузки фрагментов в viewpager и TabMenu, поэтому я сделал и рекомендую сделать это, чтобы проверить, виден ли фрагмент пользователю и если они были, получить данные и другие вещи, так вот мой код:

public class Fragment1 extends Fragment {
boolean visible = false;
public static Fragment1 newInstance() {
    return new Fragment1();
}
@Override
public View onCreateView(LayoutInflater inflater,
                         ViewGroup container, Bundle savedInstanceState) {
    if (visible && isResumed()) {
        onResume();
    } else if (visible) {
        getMyData();
    }
    return rootView;
}
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    visible = isVisibleToUser;
    if (isVisibleToUser) {
        getMyData();
    }
    else {
    }
}

@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);
}
}

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

надеюсь, это поможет.

...