Activity和Fragment相互通信或者改变UI

来源:互联网 发布:linux proc 编辑:程序博客网 时间:2024/06/03 13:36

      转载请标明原地址:http://blog.csdn.net/gaolei1201/article/details/44037831 

      在工作中我们经常会遇到Activity和Fragment交互。具体为什么要用Fragment,相信大家都懂的,可以使UI更灵活,还可以把不同模块独立化,降低耦合度。Activity和Fragment之间传值还是比较常见,但是要互相改变彼此UI还是有一定技巧的。下面根据自己工作经验,详细总结一下Activity和Fragment的几种交互方式。


一、getActivity()、new Fragment()方式( 这种方法比较经典,简单实用

下面是Activity代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1.  public class MainActivity extends FragmentActivity {  
  2.   
  3.     private TextView show_change_text;  
  4.     private Button change_fragment_bt;  
  5.     private HomeFragment homeFragment;  
  6.     @Override  
  7.     protected void onCreate(Bundle savedInstanceState) {  
  8.         super.onCreate(savedInstanceState);  
  9.         setContentView(R.layout.activity_main);  
  10.           
  11.         homeFragment=new HomeFragment();  
  12.         getSupportFragmentManager().beginTransaction()  
  13.                 .replace(R.id.fragment_content,homeFragment ).commit();  
  14.         change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);  
  15.         show_change_text = (TextView) findViewById(R.id.show_change_text);  
  16.         change_fragment_bt.setOnClickListener(new OnClickListener() {  
  17.             @Override  
  18.             public void onClick(View arg0) {  
  19.                 // TODO Auto-generated method stub  
  20.                 homeFragment.changeText();  
  21.             }  
  22.         });  
  23.     }  
  24.     public void changeText(){  
  25.         show_change_text.setText("I am UI of activity,and I have changed");  
  26.     }  
  27.   
  28.     public void onAttachFragment(Fragment fragment) {  
  29.         super.onAttachFragment(fragment);  
  30.     }  
  31.   
  32. }

下面是Fragment代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class HomeFragment extends Fragment {  
  2.     private TextView show_change_text;  
  3.     private Button change_activity_bt;  
  4.       
  5.   
  6.     public View onCreateView(LayoutInflater inflater, ViewGroup container,  
  7.             Bundle savedInstanceState) {  
  8.         super.onCreateView(inflater, container, savedInstanceState);  
  9.   
  10.         View view = inflater.inflate(R.layout.home_fragment, null);  
  11.         change_activity_bt = (Button) view  
  12.                 .findViewById(R.id.change_activity_bt);  
  13.         show_change_text = (TextView) view.findViewById(R.id.show_change_text);  
  14.         change_activity_bt.setOnClickListener(new OnClickListener() {  
  15.             @Override  
  16.             public void onClick(View arg0) {  
  17.                 // TODO Auto-generated method stub  
  18.                 ((MainActivity) getActivity()).changeText();  
  19.             }  
  20.         });  
  21.   
  22.         return view;  
  23.     }  
  24.     public void changeText(){  
  25.         show_change_text.setText("I am UI of fragment,and I have changed");  
  26.     }  
  27.   
  28.     public void onAttach(Activity activity) {  
  29.         super.onAttach(activity);  
  30.     }  
  31. }

   

   getActivity()、new Fragment()方式源码下载地址.........................

二、Handler方式


传递数据是比较容易实现,但是要用到Activity和Fragment之间互相改变彼此的UI,网上关于这个问题的解决方法还是很少的,我也没遇到合适的解决方案,自己在前人的基础上研究后创造了一种方法。使Activity和Fragment共享一个Handler,然后发送消息通知彼此更新。别的也不多说了,例子代码如下,

这是Activity的代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class MainActivity extends FragmentActivity implements OnClickListener {  
  2. private Handler mHandler;  
  3. private long lastPressBack;  
  4. private boolean isFirstPressBack = true;  
  5. private Button change_fragment_bt;  
  6. private TextView show_change_text;  
  7. private HomeFragment homeFragment;  
  8.   
  9. Handler handler = new Handler() {  
  10.     @Override  
  11.     public void handleMessage(Message msg) {  
  12.         switch (msg.what) {  
  13.         case 0:  
  14.             show_change_text.setText("I am UI of activity,and I have changed");  
  15.             break;  
  16.         }  
  17.     }  
  18. };  
  19.     @Override  
  20.     protected void onCreate(Bundle savedInstanceState) {  
  21.         super.onCreate(savedInstanceState);  
  22.         setContentView(R.layout.activity_main);  
  23.         change_fragment_bt=(Button)findViewById(R.id.change_fragment_bt);  
  24.         show_change_text=(TextView)findViewById(R.id.show_change_text);  
  25.         change_fragment_bt.setOnClickListener(this);  
  26.           
  27.         getSupportFragmentManager().beginTransaction()  
  28.         .replace(R.id.fragment_content, new HomeFragment())  
  29.         .commit();  
  30.     }  
  31.     public void onAttachFragment(Fragment fragment) {  
  32.         super.onAttachFragment(fragment);  
  33.         homeFragment=(HomeFragment) fragment;  
  34.         homeFragment.setHandler(handler) ;  
  35.     }  
  36.   
  37.     public boolean onKeyDown(int keyCode, KeyEvent event) {  
  38.         super.onKeyDown(keyCode, event);  
  39.         //点击和Fragment交互,改变Fragment的UI  
  40.         if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) {  
  41.             if(HomeFragment.changeText){  
  42.             Message msg = mHandler.obtainMessage();  
  43.                 msg.what =0;  
  44.                 mHandler.sendMessage(msg);  
  45.                 HomeFragment.changeText=false;  
  46.                 return false;  
  47.             }  
  48.             if (!isFirstPressBack) {  
  49.                 if (System.currentTimeMillis() - lastPressBack > 2 * 1000) {  
  50.                     lastPressBack = System.currentTimeMillis();  
  51.                 } else {  
  52.                     finish();  
  53.                 }  
  54.             }  
  55.             lastPressBack = System.currentTimeMillis();  
  56.             Toast.makeText(this,getString(R.string.exit_app),Toast.LENGTH_SHORT).show();  
  57.             isFirstPressBack = false;  
  58.                 return false;  
  59.             }  
  60.         return false;  
  61.         }  
  62.     public void setHandler(Handler handler) {  
  63.         mHandler = handler;  
  64.     }  
  65.     @Override  
  66.     public void onClick(View arg0) {  
  67.         // TODO Auto-generated method stub  
  68.         //点击和Fragment交互,改变Fragment的UI  
  69.         Message msg = mHandler.obtainMessage();  
  70.         msg.what =0;  
  71.         mHandler.sendMessage(msg);  
  72.         HomeFragment.changeText=false;  
  73.     }  
  74. }  
  75.  



这是Fragment的代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class HomeFragment extends Fragment implements OnClickListener{  
  2.     private TextView show_change_text;  
  3.     private Button change_activity_bt;  
  4.     private MainActivity mActivity;  
  5.     public static boolean changeText=true;  
  6.     Handler handler = new Handler() {  
  7.         @Override  
  8.         public void handleMessage(Message msg) {  
  9.             switch (msg.what) {  
  10.             case 0:  
  11.                 show_change_text.setText("I am UI of fragment,and I have changed");  
  12.                 break;  
  13.             }  
  14.         }  
  15.     };  
  16.     public View onCreateView(LayoutInflater inflater, ViewGroup container,  
  17.             Bundle savedInstanceState) {  
  18.         super.onCreateView(inflater, container, savedInstanceState);  
  19.   
  20.         View view = inflater.inflate(R.layout.home_fragment, null);  
  21.         show_change_text=(TextView)view.findViewById(R.id.show_change_text);  
  22.         change_activity_bt=(Button)view.findViewById(R.id.change_activity_bt);  
  23.         change_activity_bt.setOnClickListener(this);  
  24.         return view;  
  25.     }  
  26.     public void onAttach(Activity activity) {  
  27.         super.onAttach(activity);  
  28.         mActivity = (MainActivity) activity;  
  29.         mActivity.setHandler(handler);  
  30.     }  
  31.       
  32.      public void setHandler(Handler handler) {  
  33.             this.handler = handler;  
  34.         }  
  35.       
  36.     public void onDestroy(){  
  37.         super.onDestroy();  
  38.         changeText=true;  
  39.     }  
  40.     @Override  
  41.     public void onClick(View arg0) {  
  42.         // TODO Auto-generated method stub  
  43.         //点击和Activity交互,改变ActivityUI  
  44.         Message msg = handler.obtainMessage();  
  45.         msg.what =0;  
  46.         handler.sendMessage(msg);  
  47.     }  
  48. }  
  49.  


 Handler方式源码下载地址....................


三、接口回调方式(这个尽量不要用,因为会创建static对象,而static对象不会被回收,很占内存)


接口真是个好东西,条件是两个类,一个接口,两个类之间交互。任何两个类之间交互都可以用接口回调,针对接口编程能降低两个类之间的耦合度。


下面是Activity的代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class MainActivity extends FragmentActivity implements OnActivityChangeListener{  
  2.   
  3.     private TextView show_change_text;  
  4.     private Button change_fragment_bt;  
  5.     public static OnFragmentChangeListener onFragmentChangeListener;  
  6.     public static void setOnFragmentChangeListener(  
  7.             OnFragmentChangeListener onFragmentChangeListener) {  
  8.         MainActivity.onFragmentChangeListener = onFragmentChangeListener;  
  9.     }  
  10.     @Override  
  11.     protected void onCreate(Bundle savedInstanceState) {  
  12.         super.onCreate(savedInstanceState);  
  13.         setContentView(R.layout.activity_main);  
  14.         getSupportFragmentManager().beginTransaction()  
  15.                 .replace(R.id.fragment_content, new HomeFragment()).commit();  
  16.         change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);  
  17.         show_change_text = (TextView) findViewById(R.id.show_change_text);  
  18.         // new HomeFragment().setOnActivityChangeListener(this);  
  19.         // 这样set是不行的,因为这样又重新创造了一个HomeFragment,和原来初始化的那个不是一个,回报空指针,因为HomeFragment的listener没有被set  
  20.         //所以应该这样  HomeFragment.setOnActivityChangeListener(this);  
  21.         change_fragment_bt.setOnClickListener(new OnClickListener() {  
  22.             @Override  
  23.             public void onClick(View arg0) {  
  24.                 // TODO Auto-generated method stub  
  25.                 //在这里执行接口方法  
  26.                 onFragmentChangeListener.onFragmentChange();  
  27.             }  
  28.         });  
  29.     }  
  30.   
  31.     public void onAttachFragment(Fragment fragment) {  
  32.         super.onAttachFragment(fragment);  
  33.         // 除了这种方式传递子类对象,还可以在Fragment中  
  34.         // setOnFragmentChangeListener,可以去那里面看一下,我已经注释过  
  35.         onFragmentChangeListener = (OnFragmentChangeListener) fragment;  
  36.     }  
  37.   
  38.     @Override  
  39.     public void onActivityChange() {  
  40.         // TODO Auto-generated method stub  
  41.         // 在子类中实现接口的方法  
  42.         show_change_text.setText("I am UI of activity,and I have changed");  
  43.     }  
  44.   
  45. }

显示Fragment的代码:


[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class HomeFragment extends Fragment implements OnFragmentChangeListener {  
  2.     private TextView show_change_text;  
  3.     private Button change_activity_bt;  
  4.     public static OnActivityChangeListener onActivityChangeListener;  
  5.     public void setOnActivityChangeListener(OnActivityChangeListener onActivityChangeListener){  
  6.         HomeFragment.onActivityChangeListener=onActivityChangeListener;  
  7.     }  
  8.   
  9.     public View onCreateView(LayoutInflater inflater, ViewGroup container,  
  10.             Bundle savedInstanceState) {  
  11.         super.onCreateView(inflater, container, savedInstanceState);  
  12.   
  13.         View view = inflater.inflate(R.layout.home_fragment, null);  
  14.         change_activity_bt = (Button) view  
  15.                 .findViewById(R.id.change_activity_bt);  
  16.         show_change_text = (TextView) view.findViewById(R.id.show_change_text);  
  17.         // new MainActivity().setOnFragmentChangeListener(this);  
  18.         // 这样set是不行的,因为这样又重新创造了一个MainActivity,和原来初始化的那个不是一个,回报空指针,因为activity的listener没有被set  
  19.         //所以应该这样  MainActivity.setOnFragmentChangeListener(this);  
  20.         change_activity_bt.setOnClickListener(new OnClickListener() {  
  21.             @Override  
  22.             public void onClick(View arg0) {  
  23.                 // TODO Auto-generated method stub  
  24.                 //在这里执行接口方法  
  25.                 onActivityChangeListener.onActivityChange();  
  26.             }  
  27.         });  
  28.   
  29.         return view;  
  30.     }  
  31.   
  32.     public void onAttach(Activity activity) {  
  33.         super.onAttach(activity);  
  34.         // 除了这种方式传递子类对象,还可以在Activity中  
  35.                 // setOnActivityChangeListener,可以去那里面看一下,我已经注释过  
  36.         onActivityChangeListener=(OnActivityChangeListener) activity;  
  37.     }  
  38.   
  39.     @Override  
  40.     public void onFragmentChange() {  
  41.         // 在子类中实现接口的方法  
  42.         show_change_text.setText("I am UI of fragment,and I have changed");  
  43.     }  
  44. }  
  45.   

       Interface方式代码下载地址.....................

四、使用startActivityForResult(Fragment和Activity、两个Activity之间交互都行

下面是Fragment的代码

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class HomeFragment extends Fragment {  
  2.     private TextView fragment_text;  
  3.     private Button change_activity_bt;  
  4.   
  5.     public View onCreateView(LayoutInflater inflater, ViewGroup container,  
  6.             Bundle savedInstanceState) {  
  7.         super.onCreateView(inflater, container, savedInstanceState);  
  8.   
  9.         View view = inflater.inflate(R.layout.home_fragment, null);  
  10.         change_activity_bt = (Button) view  
  11.                 .findViewById(R.id.change_activity_bt);  
  12.         fragment_text = (TextView) view.findViewById(R.id.fragment_text);  
  13.   
  14.         change_activity_bt.setOnClickListener(new OnClickListener() {  
  15.             @Override  
  16.             public void onClick(View arg0) {  
  17.                 // TODO Auto-generated method stub  
  18.                 Intent intent = new Intent(getActivity(), SecondActivity.class);  
  19.                 startActivityForResult(intent, 0);  
  20.             }  
  21.         });  
  22.         return view;  
  23.     }  
  24.   
  25.     public void onActivityResult(int requestCode, int resultCode, Intent intent) {  
  26.         if (intent == null)  
  27.             return;  
  28.         String change01 = intent.getStringExtra("change01");  
  29.         // 根据上面发送过去的请求吗来区别  
  30.         switch (requestCode) {  
  31.         case 0:  
  32.             fragment_text.setText(change01);  
  33.             break;  
  34.         default:  
  35.             break;  
  36.         }  
  37.     }  
  38. }

下面是SecondActivity代码

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class SecondActivity extends FragmentActivity {  
  2.   
  3.     private Button change_fragment_bt;  
  4.     @Override  
  5.     protected void onCreate(Bundle savedInstanceState) {  
  6.         super.onCreate(savedInstanceState);  
  7.         setContentView(R.layout.second_activity);  
  8.         change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);  
  9.         change_fragment_bt.setOnClickListener(new OnClickListener() {  
  10.             @Override  
  11.             public void onClick(View arg0) {  
  12.                 // TODO Auto-generated method stub  
  13.                 Intent mIntent = new Intent();  
  14.                 mIntent.putExtra("change01", "Change HomeFragment By SecondActivity");  
  15.                 mIntent.putExtra("change02", "Change MainActivity By SecondActivity");  
  16.                 // 设置结果,并进行传送  
  17.                 setResult(0, mIntent);  
  18.                 finish();  
  19.             }  
  20.         });  
  21.     }  
  22. } 

stratActivityForResult方式源码,点击下载......


五、万能的BroadCastReceiver(Fragment和Activity、两个Activity之间交互都行)

虽然广播比较万能,但是它是由系统决定,且有优先级,可能会造成延迟。

下面是MainActivity 代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class MainActivity extends Activity {    
  2.     
  3.     TextView textView;    
  4.     Button sButton;    
  5.     
  6.     @Override    
  7.     protected void onCreate(Bundle savedInstanceState) {    
  8.         super.onCreate(savedInstanceState);    
  9.         setContentView(R.layout.activity_main);    
  10.         textView = (TextView) findViewById(R.id.show);    
  11.         sButton = (Button) findViewById(R.id.startAnother);    
  12.         sButton.setOnClickListener(new OnClickListener() {    
  13.                 
  14.             @Override    
  15.             public void onClick(View v) {    
  16.                 // TODO Auto-generated method stub    
  17.                 startActivity(new Intent(MainActivity.this,AnotherActivity.class));    
  18.             }    
  19.         });    
  20.         IntentFilter filter = new IntentFilter(AnotherActivity.action);    
  21.         registerReceiver(broadcastReceiver, filter);    
  22.     }    
  23.     
  24.     BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {    
  25.     
  26.         @Override    
  27.         public void onReceive(Context context, Intent intent) {    
  28.             // TODO Auto-generated method stub    
  29.             textView.setText(intent.getExtras().getString("data"));    
  30.         }    
  31.     };    
  32.     
  33.     protected void onDestroy() {    
  34.         unregisterReceiver(broadcastReceiver);    
  35.     };    
  36.  
下面是AnotherActivity代码:

[html] view plain copy
print?在CODE上查看代码片派生到我的代码片
  1. public class AnotherActivity extends Activity {    
  2.     
  3.     public static final String action = "jason.broadcast.action";    
  4.     
  5.     Button update;    
  6.     
  7.     @Override    
  8.     protected void onCreate(Bundle savedInstanceState) {    
  9.         // TODO Auto-generated method stub    
  10.         super.onCreate(savedInstanceState);    
  11.         setContentView(R.layout.another);    
  12.         update = (Button) findViewById(R.id.updateMain);    
  13.         update.setOnClickListener(new OnClickListener() {    
  14.     
  15.             @Override    
  16.             public void onClick(View v) {    
  17.                 // TODO Auto-generated method stub    
  18.                 Intent intent = new Intent(action);    
  19.                 intent.putExtra("data", "yes i am data");    
  20.                 sendBroadcast(intent);    
  21.                 finish();    
  22.             }    
  23.         });    
  24.     }    
  25. }  



0 0
原创粉丝点击