欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Android中Fragment和Activity之间的通信

程序员文章站 2022-03-04 19:42:46
...

一、接口回调的简单引入:
我们先来看一个简单的接口回调的例子:

新建一个Java工程,然后新建一个包。然后新建一个A.java文件:

A.java代码如下:

 1 package com.cn.callback;
 2 
 3 public class A {
 4     public A() {
 5         
 6     }
 7     
 8     //下载图片的操作
 9     public void loadImage(String image_path,final CallBack callBack) {
10         new Thread(new Runnable(){
11 
12             public void run() {
13                 // TODO Auto-generated method stub
14                 String msg = "Hello world";
15                 callBack.getResult(msg);                
16             }
17             
18         }).start();
19     }
20     
21     public interface CallBack {
22         public void getResult(String result);
23     }
24 } 

第21至23行就是回调方法。

新建B.java,代码如下:
1 package com.cn.callback;
2
3 import com.cn.callback.A.CallBack;
4
5 public class B {
6 public B(){
7
8 }
9
10 public static void main(String args[]) {
11 A a = new A();
12 a.loadImage(“http://www.baidu.com/a.gif“, new CallBack() {
13 public void getResult(String result) {
14 // TODO Auto-generated method stub
15 System.out.println(result);
16 }
17
18 });
19 }
20 }

Android中Fragment和Activity之间的通信

关于接口回调,有一个博客,不过现在还不能完全理解,附上链接:

一个经典例子让你彻彻底底理解java回调机制:http://blog.csdn.net/xiaanming/article/details/8703708

二、Fragment和Activity的交互:

1、在Fragment中调用Activity中的方法:

Fragment可以通过getActivity()方法来获得Activity的实例,然后就可以调用一些例如findViewById()之类的方法。例如:

View listView = getActivity().findViewById(R.id.list);
但是注意调用getActivity()时,fragment必须和activity关联(attached to an activity),否则将会返回一个null。

另外,当碎片中需要使用Context对象时,也可以使用getActivity()方法,因此获取到的活动本身就是一个Context对象。

【实例】在Activity的EditText中输入一段文本,这个时候,点击Fragment中的按钮,让它弹出吐司,显示出对应的文本。

其实就是让Activity中的文本显示在Fragment中,Fragment的核心代码如下:

 1     public View onCreateView(LayoutInflater inflater, ViewGroup container,
 2             Bundle savedInstanceState) {
 3         View view = inflater.inflate(R.layout.fragment_left, null);    
 4         button = (Button) view.findViewById(R.id.button1);
 5         button.setOnClickListener(new OnClickListener() {            
 6             @Override
 7             public void onClick(View v) {
 8                 // TODO Auto-generated method stub
 9                 EditText editText = (EditText) getActivity().findViewById(R.id.editText);
10                 Toast.makeText(getActivity(), editText.getText().toString(), 1).show();
11             }
12         });
13 
14         return view;
15     }

第09行代码是核心,通过getActivity()方法来获得Activity的实例,然后就可以调用findViewById()的方法得到其中的EditText控件。

2、在Activity中调用Fragment中的方法:(要用到接口回调)

activity也可以获得一个fragment的引用,从而调用fragment中的方法。获得fragment的引用要用FragmentManager,之后可以调用findFragmentById() 或者 findFragmentByTag()。例如:

ExampleFragment fragment = (ExampleFragment) getFragmentManager().findFragmentById(R.id.example_fragment);

具体例子稍后再讲。

3、Fragment与Fragment之间的通信:

既然Fragment和Activity之间的通信问题解决了,那Fragment与Fragment之间的通信也没有那么复杂。基本思路是:

首先在一个Fragment中可以得到与它相关联的Activity,然后再通过这个Activity去获取另外一个Fragment的实例,这样就实现了不同Fragment之间的通信。

三、创建事件回调(在Activity中获取Fragment中的值):

一些情况下,可能需要fragment和activity共享事件,一个比较好的做法是在fragment里面定义一个回调接口,然后要求宿主activity实现这个接口。当activity通过这个接口接收到一个回调,它可以让同布局中的其他fragment分享这个信息。

例如,一个新闻显示应用在一个activity中有两个fragment,一个fragment A显示文章题目的列表,一个fragment B显示文章。所以当一个文章被选择的时候,fragment A必须通知activity,然后activity通知fragment B,让它显示这篇文章。(例子的代码见官方文档)

我们现在举一个其他的例子:

【实例】在Fragment中输入值,点击Activity中的按钮,弹出吐司,显示之前输入的值。其实就是让Fragment中的文本显示在Activity中

我们在平板的左侧加入一个fragment,完整代码如下:

fragment_left.xml代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <EditText
        android:id="@+id/editText1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:ems="10" >
        <requestFocus />
    </EditText>

</LinearLayout>

其实就是加了一个EditText,方便在里面输入文本内容。

然后在frament中加一个接口回调,让它在Activity当中调用,方便获取输入文本的值。LeftFragment.java的代码如下:

 1 package com.example.m01_fragment05;
 2 
 3 import android.app.Fragment;
 4 import android.os.Bundle;
 5 import android.view.LayoutInflater;
 6 import android.view.View;
 7 import android.view.ViewGroup;
 8 import android.widget.Button;
 9 import android.widget.EditText;
10 
11 public class LeftFragment extends Fragment {
12     
13     private Button button;
14     private EditText editText;
15     
16     @Override
17     public void onCreate(Bundle savedInstanceState) {
18         super.onCreate(savedInstanceState);
19     }
20     
21     @Override
22     public View onCreateView(LayoutInflater inflater, ViewGroup container,
23             Bundle savedInstanceState) {
24         View view = inflater.inflate(R.layout.fragment_left, null);
25         editText = (EditText) view.findViewById(R.id.editText1);
26         return view;
27     }
28     
29     @Override
30     public void onPause() {
31         super.onPause();
32     }
33     
34     //接口回调
35     public void getEditText(CallBack callBack) {
36         String msg = editText.getText().toString();
37         callBack.getResult(msg);
38     }
39 
40     public interface CallBack {
41         public void getResult(String result);
42     }
43 }

代码解释如下:

第25行:一定要为editText加一个id,不然会报空指针异常的错误;

34至42行:添加一个接口回调,用于获取文本的值,然后稍后再Activity当中进行调用。

activity_main.xml的代码如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="horizontal"
    tools:context=".MainActivity" >

     <LinearLayout
        android:id="@+id/left"
        android:layout_width="224dp"
        android:layout_height="match_parent"
        android:background="#CCCCCC"
        android:orientation="vertical" >
    </LinearLayout>

    <LinearLayout
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:orientation="vertical" >

        <EditText
            android:id="@+id/editText1"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:ems="10" >
            <requestFocus />
        </EditText>

        <Button
            android:id="@+id/button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="获得Fragment的值" />
    </LinearLayout> 

</LinearLayout>

其实一共就两个线性布局,左边的现性布局留给fragment,右边的线性性局留给Activity。

MainActivity.java的代码如下:

 1 package com.example.m01_fragment05;
 2 
 3 import com.example.m01_fragment05.LeftFragment.CallBack;
 4 
 5 import android.app.Activity;
 6 import android.app.FragmentManager;
 7 import android.app.FragmentTransaction;
 8 import android.os.Bundle;
 9 import android.view.Menu;
10 import android.view.View;
11 import android.view.View.OnClickListener;
12 import android.widget.Button;
13 import android.widget.Toast;
14 
15 public class MainActivity extends Activity {
16     private FragmentManager manager;
17     private FragmentTransaction transaction;
18     private Button button;
19     @Override
20     protected void onCreate(Bundle savedInstanceState) {
21         super.onCreate(savedInstanceState);
22         setContentView(R.layout.activity_main);        
23         button = (Button)findViewById(R.id.button);
24         
25         //动态加载leftFragment
26         manager = getFragmentManager();
27         transaction = manager.beginTransaction();
28         final LeftFragment leftFragment = new LeftFragment();
29         transaction.add(R.id.left, leftFragment, "left");
30         transaction.commit();
31         button.setOnClickListener(new OnClickListener() {
32             
33             @Override
34             public void onClick(View v) {
35                 //点击按钮后,通过接口回调,获取fragment当中EditText的值,并弹出吐司
36                 leftFragment.getEditText(new CallBack(){
37                     @Override
38                     public void getResult(String result) {
39                         // TODO Auto-generated method stub
40                         Toast.makeText(MainActivity.this, result, 1).show();
41                     }                    
42                 });
43             }
44         });
45     }
46 
47     @Override
48     public boolean onCreateOptionsMenu(Menu menu) {
49         // Inflate the menu; this adds items to the action bar if it is present.
50         getMenuInflater().inflate(R.menu.main, menu);
51         return true;
52     }    
53 }

我们在Activity当中动态加载Fragment,然后点击按钮,通过接口回调,获取fragment当中EditText的值,并弹出吐司。

程序运行后,在左侧的Fragment的EditText当中输入值,点击右侧的按钮,弹出吐司,效果如下:
Android中Fragment和Activity之间的通信

相关标签: Android