一、前言

<font face = 黑体>在 Fragment 的创建、替换与移除 我们已经讲了 Fragment 的静态添加和动态添加,今天我们来讲 Fragment 的生命周期Fragment 与 Activity 之间的通信

二、Fragment 的生命周期

<font face = 黑体>和 Activity 一样,Fragment 也有自己的生命周期,并且和 Activity 的生命周期非常相似。具体如下图所示:

Fragment 生命周期
<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 替换时,生命周期的调用状况:

生命周期2

三、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


Maenj_Ba_lah
28 声望7 粉丝

真正的大师,永远怀着一颗学徒的心。