Fragment 与 Activity 之间的通信

一、前言

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

二、Fragment 的生命周期

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

Fragment 与 Activity 之间的通信_第1张图片
这里我们选择几个比较重要的状态来讲解一下:

  • onAttach():当 Fragment 和 Activity 建立关联时调用;
  • onCreateView():为 Fragment 创建视图(加载布局)时调用;
  • onActivityCreated():确保与 Fragment 相关联的 Activity 已经创建完毕时调用;
  • onDestroyView():当与 Fragment 关联的视图被移除时调用;
  • onDetach():当 Fragment 和 Activity 解除关联时调用。

2.1、体验 Fragment 的生命周期

实例一:FirstFragment 处于运行状态时生命周期的调用状况:

Fragment 与 Activity 之间的通信_第2张图片

三、Fragment 与 Activity 之间的通信

3.1、Activity 访问所属的 Fragment

实例三:在 Activity 中根据 checkBox 状态去显示 Fragment 中文字的变化,具体效果如下所示:

Fragment 与 Activity 之间的通信_第3张图片

3.1.1、访问静态添加的 Fragment

具体代码如下所示(完整代码文末给出):

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("不是程序员");
                    }
                }
            }
        });
    }
}
从代码中可以看出,在 Activity 中可以通过 getSupportFragmentManager().findFragmentById() 方法去获取到 Fragment 实例,然后获取到 Fragment 中的 View。但是这种方法只能去获取 Fragment 是通过 xml 标签添加到 Activity 里面的。

3.1.2、访问动态添加的 Fragment

具体代码如下所示(完整代码文末给出):

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("不是程序员");
                    }
                }
            }
        });
    }
}
这种情况更简单,因为我们在动态添加 Fragment 的时候,已经实例化过 Fragment 了,所以直接用就可以了。

3.2、Fragment 访问所属的 Activity

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

Fragment 与 Activity 之间的通信_第4张图片
具体代码如下所示(完整代码文末给出):

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 的控件。

四、小结

虽然通过上面两个例子我们已经实现了在 Fragment 访问 Activity,但是这种写法的耦合性是非常高的,因为我们把所有的代码都在 Fragment 中执行了,正确的做法应该是把 Fragment 只当做一个发起者,而具体的代码应该在 Activity 中执行。所以 Fragment 和 Activity 之间的最佳通信方式如下所示:

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

Fragment 与 Activity 之间的最佳通信方式 请见这篇博文。

五、源码

源码已经上传至 github

你可能感兴趣的:(android)