一、前言

<font face = 黑体>在 Fragment 的创立、替换与移除 咱们曾经讲了 Fragment 的动态增加和动静增加,明天咱们来讲 Fragment 的生命周期Fragment 与 Activity 之间的通信

二、Fragment 的生命周期

<font face = 黑体>和 Activity 一样,Fragment 也有本人的生命周期,并且和 Activity 的生命周期十分类似。具体如下图所示:


<font face = 黑体>这里咱们抉择几个比拟重要的状态来解说一下:

  • <font face = 黑体>onAttach():当 Fragment 和 Activity 建设关联时调用;
  • <font face = 黑体>onCreateView():为 Fragment 创立视图(加载布局)时调用;
  • <font face = 黑体>onActivityCreated():确保与 Fragment 相关联的 Activity 曾经创立结束时调用;
  • <font face = 黑体>onDestroyView():当与 Fragment 关联的视图被移除时调用;
  • <font face = 黑体>onDetach():当 Fragment 和 Activity 解除关联时调用。

2.1、体验 Fragment 的生命周期

<font face = 黑体>实例一:FirstFragment 处于运行状态时生命周期的调用情况:


<font face = 黑体>实例二:当 FirstFragment 被 SecondFragment 替换时,生命周期的调用情况:

三、Fragment 与 Activity 之间的通信

3.1、Activity 拜访所属的 Fragment

<font face = 黑体>实例三:在 Activity 中依据 checkBox 状态去显示 Fragment 中文字的变动,具体成果如下所示:



3.1.1、拜访动态增加的 Fragment

<font face = 黑体>具体代码如下所示(残缺代码文末给出):

public class Index3Activity extends AppCompatActivity {    private CheckBox cbIsEngineer;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_index3);        initView();    }    private void initView() {        cbIsEngineer = findViewById(R.id.cb_is_engineer);        // 在 Activity 中取得所属的 Fragment        final CheckFragment fragment = (CheckFragment) getSupportFragmentManager().findFragmentById(R.id.ft_bottom);        cbIsEngineer.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {            @Override            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {                if (isChecked) {                    if (fragment != null) {                        // Fragment 获取它的 UI 控件                        TextView tvResult = fragment.getView().findViewById(R.id.tv_result);                        tvResult.setText("是程序员");                    }                } else {                    if (fragment != null) {                        TextView tvResult = fragment.getView().findViewById(R.id.tv_result);                        tvResult.setText("不是程序员");                    }                }            }        });    }}
<font face = 黑体>从代码中能够看出,在 Activity 中能够通过 getSupportFragmentManager().findFragmentById() 办法去获取到 Fragment 实例,而后获取到 Fragment 中的 View。<font color= red>然而这种办法只能去获取 Fragment 是通过 xml 标签增加到 Activity 外面的。

3.1.2、拜访动静增加的 Fragment

<font face = 黑体>具体代码如下所示(残缺代码文末给出):

public class Index3Activity extends AppCompatActivity {    private CheckBox cbIsEngineer;    private CheckFragment fragment;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_index3);        fragment = new CheckFragment();        getSupportFragmentManager().beginTransaction().add(R.id.fragment_container, fragment).commit();        initView();    }    private void initView() {        cbIsEngineer = findViewById(R.id.cb_is_engineer);        cbIsEngineer.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {            @Override            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {                if (isChecked) {                    if (fragment != null) {                        TextView tvResult = fragment.getView().findViewById(R.id.tv_result);                        tvResult.setText("是程序员");                    }                } else {                    if (fragment != null) {                        TextView tvResult = fragment.getView().findViewById(R.id.tv_result);                        tvResult.setText("不是程序员");                    }                }            }        });    }}
<font face = 黑体>这种状况更简略,因为咱们在动静增加 Fragment 的时候,曾经实例化过 Fragment 了,所以间接用就能够了。

3.2、Fragment 拜访所属的 Activity

<font face = 黑体>实例四:在 Fragment 中点击按钮,获取 Activity 中的 checkBox 的状态,去弹出相应的 Toast,具体成果如下所示:


<font face = 黑体>具体代码如下所示(残缺代码文末给出):

public class CheckFragment extends Fragment {    private Button btnJudge;    private Toast mToast;    @Nullable    @Override    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {        View view = inflater.inflate(R.layout.fragment_check, container, false);        btnJudge = view.findViewById(R.id.btn_judge);        return view;    }    @Override    public void onActivityCreated(@Nullable Bundle savedInstanceState) {        super.onActivityCreated(savedInstanceState);        // 在 Fragment 中去取得它所属的 Activity 的控件        final CheckBox cbIsEngineer = getActivity().findViewById(R.id.cb_is_engineer);        btnJudge.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                if (cbIsEngineer != null) {                    if (cbIsEngineer.isChecked()) {                        mToast = Toast.makeText(getActivity(), "checkBox 被选中了", Toast.LENGTH_SHORT);                        mToast.setGravity(Gravity.CENTER, 0, 0);                        mToast.show();                    } else {                        mToast = Toast.makeText(getActivity(), "checkBox 没有被选中", Toast.LENGTH_SHORT);                        mToast.setGravity(Gravity.CENTER, 0, 0);                        mToast.show();                    }                }            }        });    }}
从下面代码能够看出咱们能够通过 getActivity().findViewById() 获取到 Fragment 所属的 Activity 的控件。

四、小结

<font face = 黑体>尽管通过下面两个例子咱们曾经实现了在 Fragment 拜访 Activity,然而这种写法的耦合性是十分高的,因为咱们把所有的代码都在 Fragment 中执行了,正确的做法应该是把 Fragment 只当做一个发起者,而具体的代码应该在 Activity 中执行。所以 Fragment 和 Activity 之间的最佳通信形式如下所示:

  1. 在发动事件的 Fragment 中定义一个接口,接口中申明办法;
  2. 在 onAttach() 要求 Activity 实现该接口;
  3. 在 Activity 中实现该办法。

<font face = 黑体>Fragment 与 Activity 之间的最佳通信形式 请见这篇博文。

五、源码

<font face = 黑体>源码曾经上传至 github。