(未给Fragment的布局设置BackGound)
之前关于Android中Fragment的概念以及创建方式,我专门写了一篇博文《Android中Fragment的两种创建方式》,就如何创建Fragment混合布局做了详细的分析,今天就来详细说道说道Fragment与宿主Activity之间是如何实现数据交互的。
我们可以这样理解,宿主Activity中的Fragment之间要实现信息交互,就必须通过宿主Activity,Fragment之间是不可能直接实现信息交互的。
Fragment与Fragment或者说与Activity(将部分Fragment包含的的布局直接部署在Activity中)的数据交互我个人总结了两种实现方式:
(1)通过Bundle传参,从而实现Fragment之间的数据交互
(2)通过在Fragment内部定义一个回调接口,并需要宿主Activity实现它。当Activity通过接口接收到回调时,可以在必要时与布局中的其它Fagment共享信息。
在开始两种实现方式之前,先向大家介绍并演示其中一些个性化参数不同设置后的效果,之前向大家介绍过要管理宿主Activity中的Fragment,必须得到FragmentManager(碎片管理),而FragmentManager要实现对Fragment的增删改换等操作(事务),则必须启用FragmentTransaction,这里主要向大家演示添加FragmentTransaction.addToBackStack方法前后以及在未给Fragment设置BackGround的情况下使用FragmentTransaction.add与replace的不同效果,更加详细的方法介绍以及用法请大家参照API详细了解。
未设置FragmentTransaction.addToBackStack方法演示效果(打开多层后,按回退,直接退出程序):
设置FragmentTransaction.addToBackStack方法演示效果(这里没有设置监听后台栈变化的监听器进行判断处理):
貌似看不出差别,其实在按回退键时,这个是根据打开的顺序,逐个退出
未给Fragment设置BackGround的情况下使用FragmentTransaction.add的演示效果:
注意啦注意啦,这里着重声明的是未给右侧Fragment的布局设置BackGound的情况,如果设置的BackGound,那么实现效果和replace没有差别,这也是今天超时这么多的主要原因
不同的实现效果大家都看到了,我们开始演示实现代码:
我们使用的是Android中Fragment的两种创建方式中(通过java代码将fragment添加到宿主Activity中)的布局文件,布局文件代码请参考
(1)通过Bundle传参
第一步:右侧Fragment对应的java代码RightFragment.java:
1 import android.app.Fragment;
2 import android.os.Bundle;
3 import android.view.LayoutInflater;
4 import android.view.View;
5 import android.view.ViewGroup;
6 import android.widget.TextView;
7 /**
8 * Created by panchengjia on 2016/12/18.
9 */
10 public class RightFragment extends Fragment {
11 public RightFragment() {
12 }
13 /*Fragment的传参方式(通过Bundle对象来传递)
14 *采用这种传参方式可以保证用户在横竖屏切换时所
15 * 传递的参数不会丢失
16 */
17 public static RightFragment getInstance(String data){
18 RightFragment rightFragment = new RightFragment();
19 Bundle bundle = new Bundle();
20 //将需要传递的字符串以键值对的形式传入bundle
21 bundle.putString("data",data);
22 rightFragment.setArguments(bundle);
23 return rightFragment;
24 }
25 @Override
26 public void onCreate(Bundle savedInstanceState) {
27 super.onCreate(savedInstanceState);
28 }
29 @Override
30 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
31 View view =inflater.inflate(R.layout.right_layout,container,false);
32 TextView tv = (TextView) view.findViewById(R.id.tv);
33 String data = getArguments().getString("data");
34 tv.setText(data);
35 return view;
36 }
37 @Override
38 public void onPause() {
39 super.onPause();
40 }
41 }
第二步:主布局宿主Activity对应的java实现代码MainActivity.java:
1 import android.app.FragmentManager;
2 import android.app.FragmentTransaction;
3 import android.support.v7.app.AppCompatActivity;
4 import android.os.Bundle;
5 import android.view.View;
6 import android.widget.Button;
7 public class Main3Activity extends AppCompatActivity {
8 FragmentManager fragmentManager;
9 FragmentTransaction fragmentTransaction;
10 LeftFragment leftFragment;
11 Button panhouye,bikonghai;//声明leftfragment中的按钮
12 @Override
13 protected void onCreate(Bundle savedInstanceState) {
14 super.onCreate(savedInstanceState);
15 setContentView(R.layout.activity_main2);
16 //获取fragmentManager
17 fragmentManager=getFragmentManager();
18 //通过findFragmentById找到leftFragment
19 leftFragment = (LeftFragment) fragmentManager.findFragmentById(R.id.left);
20 //找到对应的导航按钮并设置点击事件
21 panhouye = (Button) leftFragment.getView().findViewById(R.id.panhouye);
22 bikonghai = (Button) leftFragment.getView().findViewById(R.id.bikonghai);
23 panhouye.setOnClickListener(new View.OnClickListener() {
24 @Override
25 public void onClick(View v) {
26 //调用方法修改rightfragment中的文本内容
27 switchButton("我是潘侯爷");
28 }
29 });
30 bikonghai.setOnClickListener(new View.OnClickListener() {
31 @Override
32 public void onClick(View v) {
33 switchButton("我是碧空海");
34 }
35 });
36 //设置打开Activity后rightfragment中默认的文本内容
37 switchButton("我是潘侯爷");
38 }
39 //定义方法填充Activity右侧的fragment,并通过传参修改文本内容
40 public void switchButton(String data){
41 fragmentManager=getFragmentManager();
42 fragmentTransaction=fragmentManager.beginTransaction();
43 //通过调用RightFragment中的getInstance方法传修改文本
44 RightFragment rightFragment =RightFragment.getInstance(data);
45 //此时使用add方法会造成右侧fragment中文本重叠(未设置BackGround时)
46 fragmentTransaction.replace(R.id.right,rightFragment);
47 fragmentTransaction.commit();
48 }
49 }
(2)接口回调
第一步:左侧fragment的java实现代码LeftFragment.java文件
本次演示通过点击左侧Fragment中的按钮点击触发与右侧Fragment的数据交互,所以需在本类中添加回调接口用于在宿主Activity中回调修改右侧文本的方法。
1 import android.app.Fragment;
2 import android.os.Bundle;
3 import android.view.LayoutInflater;
4 import android.view.View;
5 import android.view.ViewGroup;
6 import android.widget.Button;
7 /**
8 * Created by panchengjia on 2016/12/18.
9 */
10 public class LeftFragment extends Fragment implements View.OnClickListener{
11 //声明内部定义的回调接口
12 CallBackListener callBackListener;
13 //声明布局中事件触发按钮
14 Button panhouye,bikonghai;
15 @Override
16 public void onCreate(Bundle savedInstanceState) {
17 super.onCreate(savedInstanceState);
18 //通过getActivity()获取用于回调修改文本方法的接口
19 callBackListener= (CallBackListener) getActivity();
20 }
21 @Override
22 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
23 View view =inflater.inflate(R.layout.left_layout,container,false);
24 panhouye = (Button) view.findViewById(R.id.panhouye);
25 bikonghai= (Button) view.findViewById(R.id.bikonghai);
26 panhouye.setOnClickListener(this);//为按钮设置监听事件
27 bikonghai.setOnClickListener(this);
28 return view;
29 }
30 @Override
31 public void onPause() {
32 super.onPause();
33 }
34
35 @Override
36 public void onClick(View v) {
37 switch (v.getId()){
38 case R.id.panhouye:
39 callBackListener.setText("我是潘侯爷");
40 break;
41 case R.id.bikonghai:
42 callBackListener.setText("我是碧空海");
43 break;
44 }
45 }
46 //设置用于修改文本的回调接口
47 public static interface CallBackListener{
48 public void setText(String data);
49 }
50 }
第二步:右侧fragment的java实现代码RightFragment.java文件
本次演示右侧为文本显示fragment,也是点击左侧的按钮后,通过改变文本的形式体现点击事件的处理,所以必须在本Fragment类中添加文本修改的方法。
1 import android.app.Fragment;
2 import android.os.Bundle;
3 import android.support.annotation.Nullable;
4 import android.view.LayoutInflater;
5 import android.view.View;
6 import android.view.ViewGroup;
7 import android.widget.TextView;
8 /**
9 * Created by panchengjia on 2016/12/18.
10 */
11 public class RightFragment extends Fragment {
12 //声明fragment中的TextView,用于建立修改文本的方法
13 private TextView tv;
14 @Override
15 public void onCreate(Bundle savedInstanceState) {
16 super.onCreate(savedInstanceState);
17 }
18 @Override
19 public void onPause() {
20 super.onPause();
21 }
22 @Nullable
23 @Override
24 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
25 View view =inflater.inflate(R.layout.right_layout,container,false);
26 tv = (TextView) view.findViewById(R.id.tv);
27 return view;
28 }
29 //在这里设置修改自身文本的方法
30 public void setFragmentText(String data){
31 tv.setText(data);
32 }
33 }
第三步:主界面宿主Acvtivity的java实现代码MainActivity.java文件(为了接收Fragment事件回调,宿主的Activity必须实现回调接口):
1 import android.app.FragmentManager;
2 import android.app.FragmentTransaction;
3 import android.support.v7.app.AppCompatActivity;
4 import android.os.Bundle;
5 public class MainActivity extends AppCompatActivity implements LeftFragment.CallBackListener{
6 FragmentManager fragmentManager;
7 FragmentTransaction fragmentTransaction;
8 /*leftfragment已经在主布局文件中声明,
9 *这里仅需要通过代码声明部署rightFragment
10 */
11 RightFragment rightFragment;
12 @Override
13 protected void onCreate(Bundle savedInstanceState) {
14 super.onCreate(savedInstanceState);
15 setContentView(R.layout.activity_main);
16 //初始化主布局(主要目的是为主布局填充fragments)
17 initActivity();
18 }
19 private void initActivity() {
20 fragmentManager = getFragmentManager();
21 fragmentTransaction = fragmentManager.beginTransaction();
22 rightFragment = new RightFragment();
23 fragmentTransaction.add(R.id.right,rightFragment);
24 fragmentTransaction.commit();
25 }
26 //接口实现方法,用于回调RightFragment类中定义的修改文本的方法
27 @Override
28 public void setText(String data) {
29 rightFragment.setFragmentText(data);
30 }
31 }
小结:
本次演示接口回调的实现方式看起来比使用bundle传参的代码量大了一些,但在实际开发中,我们面临的Fragment不仅仅是眼前的这两个,而使用接口回调实现交互数据的方法能更好的实现重用Fragment UI组件,从根本上解决的大量代码重用的问题,建议大家熟练掌握接口回调来实现数据交互。