Activity和Fragment的几种交互方式(Activity可以改变Fragment的UI,Fragment也可以改变ActivityUI)

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

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

下面是Activity代码:

public class MainActivity extends FragmentActivity implements OnActivityChangeListener{
 
    private TextView show_change_text;
    private Button change_fragment_bt;
    public static OnFragmentChangeListener onFragmentChangeListener;
    public static void setOnFragmentChangeListener(
            OnFragmentChangeListener onFragmentChangeListener) {
        MainActivity.onFragmentChangeListener = onFragmentChangeListener;
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getSupportFragmentManager().beginTransaction()
                .replace(R.id.fragment_content, new HomeFragment()).commit();
        change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);
        show_change_text = (TextView) findViewById(R.id.show_change_text);
        // new HomeFragment().setOnActivityChangeListener(this);
        // 这样set是不行的,因为这样又重新创造了一个HomeFragment,和原来初始化的那个不是一个,回报空指针,因为HomeFragment的listener没有被set
        //所以应该这样  HomeFragment.setOnActivityChangeListener(this);
        change_fragment_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                //在这里执行接口方法
                onFragmentChangeListener.onFragmentChange();
            }
        });
    }
 
    public void onAttachFragment(Fragment fragment) {
        super.onAttachFragment(fragment);
        // 除了这种方式传递子类对象,还可以在Fragment中
        // setOnFragmentChangeListener,可以去那里面看一下,我已经注释过
        onFragmentChangeListener = (OnFragmentChangeListener) fragment;
    }
 
    @Override
    public void onActivityChange() {
        // TODO Auto-generated method stub
        // 在子类中实现接口的方法
        show_change_text.setText("I am UI of activity,and I have changed");
    }
 
}

下面是Fragment代码:

public class HomeFragment extends Fragment implements OnFragmentChangeListener {
    private TextView show_change_text;
    private Button change_activity_bt;
    public static OnActivityChangeListener onActivityChangeListener;
    public void setOnActivityChangeListener(OnActivityChangeListener onActivityChangeListener){
        HomeFragment.onActivityChangeListener=onActivityChangeListener;
    }
 
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
 
        View view = inflater.inflate(R.layout.home_fragment, null);
        change_activity_bt = (Button) view
                .findViewById(R.id.change_activity_bt);
        show_change_text = (TextView) view.findViewById(R.id.show_change_text);
        // new MainActivity().setOnFragmentChangeListener(this);
        // 这样set是不行的,因为这样又重新创造了一个MainActivity,和原来初始化的那个不是一个,回报空指针,因为activity的listener没有被set
        //所以应该这样  MainActivity.setOnFragmentChangeListener(this);
        change_activity_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                //在这里执行接口方法
                onActivityChangeListener.onActivityChange();
            }
        });
 
        return view;
    }
 
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        // 除了这种方式传递子类对象,还可以在Activity中
                // setOnActivityChangeListener,可以去那里面看一下,我已经注释过
        onActivityChangeListener=(OnActivityChangeListener) activity;
    }
 
    @Override
    public void onFragmentChange() {
        // 在子类中实现接口的方法
        show_change_text.setText("I am UI of fragment,and I have changed");
    }
}

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

二、Handler方式

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

这是Activity的代码:

public class MainActivity extends FragmentActivity implements OnClickListener {
private Handler mHandler;
private long lastPressBack;
private boolean isFirstPressBack = true;
private Button change_fragment_bt;
private TextView show_change_text;
private HomeFragment homeFragment;
 
Handler handler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
        switch (msg.what) {
        case 0:
            show_change_text.setText("I am UI of activity,and I have changed");
            break;
        }
    }
};
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        change_fragment_bt=(Button)findViewById(R.id.change_fragment_bt);
        show_change_text=(TextView)findViewById(R.id.show_change_text);
        change_fragment_bt.setOnClickListener(this);
        
        getSupportFragmentManager().beginTransaction()
        .replace(R.id.fragment_content, new HomeFragment())
        .commit();
    }
    public void onAttachFragment(Fragment fragment) {
        super.onAttachFragment(fragment);
        homeFragment=(HomeFragment) fragment;
        homeFragment.setHandler(handler) ;
    }
 
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        super.onKeyDown(keyCode, event);
        //点击和Fragment交互,改变Fragment的UI
        if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) {
            if(HomeFragment.changeText){
            Message msg = mHandler.obtainMessage();
                msg.what =0;
                mHandler.sendMessage(msg);
                HomeFragment.changeText=false;
                return false;
            }
            if (!isFirstPressBack) {
                if (System.currentTimeMillis() - lastPressBack > 2 * 1000) {
                    lastPressBack = System.currentTimeMillis();
                } else {
                    finish();
                }
            }
            lastPressBack = System.currentTimeMillis();
            Toast.makeText(this,getString(R.string.exit_app),Toast.LENGTH_SHORT).show();
            isFirstPressBack = false;
                return false;
            }
        return false;
        }
    public void setHandler(Handler handler) {
        mHandler = handler;
    }
    @Override
    public void onClick(View arg0) {
        // TODO Auto-generated method stub
        //点击和Fragment交互,改变Fragment的UI
        Message msg = mHandler.obtainMessage();
        msg.what =0;
        mHandler.sendMessage(msg);
        HomeFragment.changeText=false;
    }
}

这是Fragment的代码:

public class HomeFragment extends Fragment implements OnClickListener{
    private TextView show_change_text;
    private Button change_activity_bt;
    private MainActivity mActivity;
    public static boolean changeText=true;
    Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
            case 0:
                show_change_text.setText("I am UI of fragment,and I have changed");
                break;
            }
        }
    };
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
 
        View view = inflater.inflate(R.layout.home_fragment, null);
        show_change_text=(TextView)view.findViewById(R.id.show_change_text);
        change_activity_bt=(Button)view.findViewById(R.id.change_activity_bt);
        change_activity_bt.setOnClickListener(this);
        return view;
    }
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        mActivity = (MainActivity) activity;
        mActivity.setHandler(handler);
    }
    
     public void setHandler(Handler handler) {
            this.handler = handler;
        }
    
    public void onDestroy(){
        super.onDestroy();
        changeText=true;
    }
    @Override
    public void onClick(View arg0) {
        // TODO Auto-generated method stub
        //点击和Activity交互,改变ActivityUI
        Message msg = handler.obtainMessage();
        msg.what =0;
        handler.sendMessage(msg);
    }
}

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

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

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

下面是Activity的代码:

public class MainActivity extends FragmentActivity implements OnActivityChangeListener{
 
    private TextView show_change_text;
    private Button change_fragment_bt;
    public static OnFragmentChangeListener onFragmentChangeListener;
    public static void setOnFragmentChangeListener(
            OnFragmentChangeListener onFragmentChangeListener) {
        MainActivity.onFragmentChangeListener = onFragmentChangeListener;
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getSupportFragmentManager().beginTransaction()
                .replace(R.id.fragment_content, new HomeFragment()).commit();
        change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);
        show_change_text = (TextView) findViewById(R.id.show_change_text);
        // new HomeFragment().setOnActivityChangeListener(this);
        // 这样set是不行的,因为这样又重新创造了一个HomeFragment,和原来初始化的那个不是一个,回报空指针,因为HomeFragment的listener没有被set
        //所以应该这样  HomeFragment.setOnActivityChangeListener(this);
        change_fragment_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                //在这里执行接口方法
                onFragmentChangeListener.onFragmentChange();
            }
        });
    }
 
    public void onAttachFragment(Fragment fragment) {
        super.onAttachFragment(fragment);
        // 除了这种方式传递子类对象,还可以在Fragment中
        // setOnFragmentChangeListener,可以去那里面看一下,我已经注释过
        onFragmentChangeListener = (OnFragmentChangeListener) fragment;
    }
 
    @Override
    public void onActivityChange() {
        // TODO Auto-generated method stub
        // 在子类中实现接口的方法
        show_change_text.setText("I am UI of activity,and I have changed");
    }
 
}

显示Fragment的代码:


<span style="font-size:14px;"><span style="font-size:14px;color:#333333;">public class HomeFragment extends Fragment implements OnFragmentChangeListener {
    private TextView show_change_text;
    private Button change_activity_bt;
    public static OnActivityChangeListener onActivityChangeListener;
    public void setOnActivityChangeListener(OnActivityChangeListener onActivityChangeListener){
        HomeFragment.onActivityChangeListener=onActivityChangeListener;
    }
 
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
 
        View view = inflater.inflate(R.layout.home_fragment, null);
        change_activity_bt = (Button) view
                .findViewById(R.id.change_activity_bt);
        show_change_text = (TextView) view.findViewById(R.id.show_change_text);
        // new MainActivity().setOnFragmentChangeListener(this);
        // 这样set是不行的,因为这样又重新创造了一个MainActivity,和原来初始化的那个不是一个,回报空指针,因为activity的listener没有被set
        //所以应该这样  MainActivity.setOnFragmentChangeListener(this);
        change_activity_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                //在这里执行接口方法
                onActivityChangeListener.onActivityChange();
            }
        });
 
        return view;
    }
 
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        // 除了这种方式传递子类对象,还可以在Activity中
                // setOnActivityChangeListener,可以去那里面看一下,我已经注释过
        onActivityChangeListener=(OnActivityChangeListener) activity;
    }
 
    @Override
    public void onFragmentChange() {
        // 在子类中实现接口的方法
        show_change_text.setText("I am UI of fragment,and I have changed");
    }
}
</span></span>

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

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

下面是Fragment的代码

public class HomeFragment extends Fragment {
    private TextView fragment_text;
    private Button change_activity_bt;
 
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
 
        View view = inflater.inflate(R.layout.home_fragment, null);
        change_activity_bt = (Button) view
                .findViewById(R.id.change_activity_bt);
        fragment_text = (TextView) view.findViewById(R.id.fragment_text);
 
        change_activity_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                Intent intent = new Intent(getActivity(), SecondActivity.class);
                startActivityForResult(intent, 0);
            }
        });
        return view;
    }
 
    public void onActivityResult(int requestCode, int resultCode, Intent intent) {
        if (intent == null)
            return;
        String change01 = intent.getStringExtra("change01");
        // 根据上面发送过去的请求吗来区别
        switch (requestCode) {
        case 0:
            fragment_text.setText(change01);
            break;
        default:
            break;
        }
    }
}

下面是SecondActivity代码

<span style="font-size:14px;">public class SecondActivity extends FragmentActivity {
 
    private Button change_fragment_bt;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.second_activity);
        change_fragment_bt = (Button) findViewById(R.id.change_fragment_bt);
        change_fragment_bt.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                Intent mIntent = new Intent();
                mIntent.putExtra("change01", "Change HomeFragment By SecondActivity");
                mIntent.putExtra("change02", "Change MainActivity By SecondActivity");
                // 设置结果,并进行传送
                setResult(0, mIntent);
                finish();
            }
        });
    }
}</span>

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

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

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

下面是MainActivity 代码:

<span style="font-size:14px;">public class MainActivity extends Activity {  
  
    TextView textView;  
    Button sButton;  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_main);  
        textView = (TextView) findViewById(R.id.show);  
        sButton = (Button) findViewById(R.id.startAnother);  
        sButton.setOnClickListener(new OnClickListener() {  
              
            @Override  
            public void onClick(View v) {  
                // TODO Auto-generated method stub  
                startActivity(new Intent(MainActivity.this,AnotherActivity.class));  
            }  
        });  
        IntentFilter filter = new IntentFilter(AnotherActivity.action);  
        registerReceiver(broadcastReceiver, filter);  
    }  
  
    BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {  
  
        @Override  
        public void onReceive(Context context, Intent intent) {  
            // TODO Auto-generated method stub  
            textView.setText(intent.getExtras().getString("data"));  
        }  
    };  
  
    protected void onDestroy() {  
        unregisterReceiver(broadcastReceiver);  
    };  
} </span>

下面是AnotherActivity代码:

<span style="font-size:14px;">public class AnotherActivity extends Activity {  
  
    public static final String action = "jason.broadcast.action";  
  
    Button update;  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        // TODO Auto-generated method stub  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.another);  
        update = (Button) findViewById(R.id.updateMain);  
        update.setOnClickListener(new OnClickListener() {  
  
            @Override  
            public void onClick(View v) {  
                // TODO Auto-generated method stub  
                Intent intent = new Intent(action);  
                intent.putExtra("data", "yes i am data");  
                sendBroadcast(intent);  
                finish();  
            }  
        });  
    }  
}  </span>

六、可以使用第三方框架EventBus,可参考:http://blog.csdn.net/lmj623565791/article/details/40794879

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,425评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,058评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,186评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,848评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,249评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,554评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,830评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,536评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,239评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,505评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,004评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,346评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,999评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,060评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,821评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,574评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,480评论 2 267