Fragment 与 Activity 之间的通信

    科技2022-08-28  102

    一、前言

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

    二、Fragment 的生命周期

    和 Activity 一样,Fragment 也有自己的生命周期,并且和 Activity 的生命周期非常相似。具体如下图所示: 这里我们选择几个比较重要的状态来讲解一下:

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

    2.1、体验 Fragment 的生命周期

    实例一:FirstFragment 处于运行状态时生命周期的调用状况: 实例二:当 FirstFragment 被 SecondFragment 替换时,生命周期的调用状况:

    三、Fragment 与 Activity 之间的通信

    3.1、Activity 访问所属的 Fragment

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

    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,具体效果如下所示: 具体代码如下所示(完整代码文末给出):

    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 之间的最佳通信方式如下所示:

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

    Fragment 和 Activity 之间的最佳通信方式我们下篇博文再讲。

    五、源码

    源码已经上传至 github。

    Processed: 0.010, SQL: 9