初始化时片段的布局字段为NULL

时间:2018-05-12 06:37:51

标签: android android-fragments android-tablayout fragmentpageradapter butterknife

我遇到将数据传递给片段的问题。它在生产中的所有时间都有0.1%的崩溃。让我们说100k开放活动100次。它看起来并不常见,但它非常困扰我,我认为我在使用数据进行片段初始化时出错了。问题是,我只创建了一次片段,而其他所有时间我都需要将数据传递给它们我正在做下一步:myFragmentInstance.setData(Object someData);然后发生崩溃,因为它告诉我们片段中的那些视图元素没有找到并且它们是NULL,但如果我没有重新创建它们,一切都应该没问题。我没有旋转手机,也没有足够的内存。它发生在网络重新连接上,因为在网络重新连接时,我将服务器获取新数据,然后将新数据设置为我的片段。我有我使用的两个片段的字段照片,maby你们中的一些人知道这些数据可以告诉崩溃时片段的状态。 我正在使用库ButterKnife初始化片段和活动的字段,而不是用findById初始化它,maby它有一些影响还是没有?

以下是简单项目的链接(仅限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 extends 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>

2 个答案:

答案 0 :(得分:2)

我很确定您的问题是由于保留了对数组中片段的引用。片段具有生命周期,并且不保证引用仍然存在。正如你所说,很难重现和追踪究竟出了什么问题,但也许你不需要。 关于如何解决这个问题的一些建议:

  1. 不要存储对片段的引用。几乎按照Google页面上的示例(https://developer.android.com/training/animation/screen-slide)并在每次请求时实例化一个新片段。

  2. 如果您担心性能并且缓存正在解决它,请尝试使用FragmentStatePagerAdapter - 它会缓存页面并管理片段的状态。

  3. 如果您需要从主片段(或活动)访问页面片段,而不是存储引用,请使用`findFragmentByTag',它将始终返回片段的当前活动实例。

答案 1 :(得分:1)

我坚信事情并不像其他人所解释的那样复杂。如果我理解正确,网络交易造成的延迟是罪魁祸首。

考虑这种情况。

  • 您正在制作网络请求,在视图中进行了一些更改 最后。
  • 您切换了寻呼机。 (片段是分离的,观点是 破坏)
  • 来自网络请求的回调。你猜怎么着! 崩溃

因此,在处理碎片视图时,更加小心总是一个好主意。我经常这样做。

//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
}

或者,您也可以将潜在代码包装在try catch中。但这更像是一个盲目的镜头(至少在这种情况下)。