监听Fragment生命周期

引言

Android中,我们管理界面的主要组件是Activity和Fragment。Android为我们提供了Activity和Fragment的生命周期,让我们知道组件的运行情况。我们可以在各个生命周期中加入一些相关的业务逻辑。

但是,要在每个生命周期中加入代码非常繁琐,并且,很容易造成耦合。在软件逐渐模块化、插件化的大背景下,我们希望能够在其他模块默默地监听Fragment的生命周期。Fragment只需要在启动时注册被监听即可。

通过,这样的手段,我们就能够将Fragment生命周期中的逻辑代码进行分离。

思路

Fragment不能直接从外部监听生命周期的变化,所以我们采用一种间接的办法。

我们新建一个没有界面的Fragment,称之ListenerFragment。将ListenerFragment作为ChildFragment加入被监听的Fragment中。由于ListenerFragment没有界面,那么ListenerFragment的生命周期一定与被监听的Fragment一致。

因此我们做成了这样的结构:

当FragmentA生命周期发生变化时,ListenerFragmentA也会随之变化。

实现

ListenerFragment.java

public class ListenerFragment extends Fragment {


    public interface FragmentLifecycler{
        void onStart();
        void onStop();
        void onDestroy();
    }

    FragmentLifecycler mFragmentLifecycler;

    public void setFragmentLifecycler(FragmentLifecycler lifecycler){
        mFragmentLifecycler = lifecycler;
    }


    @Override
    public void onStart() {
        super.onStart();
        mFragmentLifecycler.onStart();

    }

    @Override
    public void onStop() {
        super.onStop();
        mFragmentLifecycler.onStop();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        mFragmentLifecycler.onDestroy();
    }
}

FragmentListener.java

public class FragmentListener {

    static final String LISTENER_FRAGMENT_TAG = "listener_fragment_tag";

    private ListenerFragment mListenerFragment;
    private Context mContext;

    /**
     * Register fragment into FragmentListener
     * @param fragment
     */
    public void registerFragment(Fragment fragment) {
        mContext = fragment.getActivity();
        if (mListenerFragment == null) {
            mListenerFragment = new ListenerFragment();
        }
        mListenerFragment.setFragmentLifecycler(new ListenerFragment.FragmentLifecycler() {
            @Override
            public void onStart() {
                Toast.makeText(mContext, "onStart", Toast.LENGTH_SHORT).show();
            }

            @Override
            public void onStop() {
                Toast.makeText(mContext, "onStop", Toast.LENGTH_SHORT).show();
            }

            @Override
            public void onDestroy() {
                Toast.makeText(mContext, "onDestroy", Toast.LENGTH_SHORT).show();
            }
        });
        // 由于Fragment的bug,必须将mChildFragmentManager的accessible设为true
        compatibleFragment(fragment);

        fragment.getChildFragmentManager()
                .beginTransaction()
                .add(mListenerFragment, LISTENER_FRAGMENT_TAG)
                .commitAllowingStateLoss();
    }

    /**
     * For bug of Fragment in Android
     * https://issuetracker.google.com/issues/36963722
     * @param fragment
     */
    private void compatibleFragment(Fragment fragment) {
        try {
            Field childFragmentManager = Fragment.class.getDeclaredField("mChildFragmentManager");
            childFragmentManager.setAccessible(true);
            childFragmentManager.set(fragment, null);
        } catch (NoSuchFieldException e) {
            throw new RuntimeException(e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }
}

ParentFragmentA.java

public class ParentFragmentA extends Fragment {

    FragmentListener mFragmentListener;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
        View view = inflater.inflate(R.layout.parent_fragment_layout, container, false);
        return view;
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        TextView textView = (TextView) view.findViewById(R.id.text_id);
        textView.setText("This is A");
        mFragmentListener = new FragmentListener();
        mFragmentListener.registerFragment(this);

    }
}

被监听的Fragment在onViewCreated中注册监听,即可。

踩坑

在这个方法中,使用到了getChildFragmentManager()方法。然而在现在的Android中存在Bug。当Fragment被replace掉之后,再replace回来。getChildFragmentManager就会出错。原因是,此时mChildFragmentManager中的mActivity为null。

为了解决这个问题,需要对被监听的Fragment进行一些修改。compatibleFragment方法就是完成这件事。

    /**
     * For bug of Fragment in Android
     * https://issuetracker.google.com/issues/36963722
     * @param fragment
     */
    private void compatibleFragment(Fragment fragment) {
        try {
            Field childFragmentManager = Fragment.class.getDeclaredField("mChildFragmentManager");
            childFragmentManager.setAccessible(true);
            childFragmentManager.set(fragment, null);
        } catch (NoSuchFieldException e) {
            throw new RuntimeException(e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }

总结

本文只提供了最简单的思路说明,在实际的工程开发中,每次在registerFragment中新建ListenerFragment显示不是个好方法,但太多的工程代码会加大理解难度,故只在此做了一个简单的示例。

欢迎指正!

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Android知识点总结

4-AVI--Fragment与ViewPager结合

1063
来自专栏佳爷的后花媛

动态创建Fragment

5.0 在使用fragment的activity里面调用getFragmentManager方法.得到fragmentManager对象

711
来自专栏CodingBlock

Android学习笔记(六)Fragment的生命周期

  在上一篇博文中对Fragment做了简单的介绍,现在再来探讨一下Fragment的生命周期。 一、Fragment的几种状态:   与Activity类...

23210
来自专栏Android知识点总结

2-VVI-材料设计之TabLayout下标签

845
来自专栏pangguoming

Android EditText 获得输入焦点 以及requestfocus()失效的问题

 最近做公司项目的时候,经常会遇到一个问题,就是我为某个控件如EditText设置requestfocus()的时候不管用,比如说登陆的时候,我判断下用户输入的...

3486
来自专栏向治洪

android的RadioGroup讲解

这个主要是如何替换fragment的demo。效果图如下(下面的tabhost和上面的bar不属于这次的内容,这个是我做的一个应用程序框架的一部分,有需要的或者...

20510
来自专栏Android机器圈

Android之Bmob移动后端云服务器

源码下载:http://download.csdn.net/download/jjhahage/10034519 PS:一般情况下,我们在写android程序的...

6709
来自专栏Android中高级开发

Android开发之漫漫长途 XII——Fragment详解

该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列。该系列引用了《Android开发艺术探索...

781
来自专栏CodingBlock

Android学习笔记(五)Fragment简介

  Fragment是在Android 3.0 (API level 11)中引入的Activity的子模块。初衷是为了适应大屏幕的平板电脑,我们只需要使用Fr...

2088
来自专栏Java后端生活

购物APP项目开发(1)——菜单开发

1314

扫码关注云+社区