1. 程式人生 > >Android中使用開源框架EventBus3.0實現Fragment之間的通訊互動

Android中使用開源框架EventBus3.0實現Fragment之間的通訊互動

1.概述

在之前的博文中簡單介紹過如何實現fragment之間的資訊互動:《Android中Fragment與Activity之間的互動(兩種實現方式)》,今天繼續給大家介紹一種可以實現此效果的另外一種方式EventBus。(相比於handler,介面回撥,bundle傳參,這個簡單好用到哭)

EventBus是Android下高效的釋出/訂閱事件的訊息匯流排。作用是可以代替傳統的Intent,Handler,Broadcast或介面函式在Fragment、Activity、Service、執行緒之間傳遞資料進行通訊,執行方法。做為訊息匯流排,有三個主要元素:

(1)Event:事件

(2)Subscriber:事件訂閱者,接受特定的事件

(3)Publisher:事件釋出者,用於通知Subscriber有事件發生

結合EventBus以上的三個元素,我們也可以稱其為一種觀察者設計模式。

EventBus 官網連結http://greenrobot.org/eventbus/

EventBus GitHub連結https://github.com/greenrobot/EventBus

前期相關博文連結:

Android中Fragment與Activity之間的互動(兩種實現方式)

Android中Fragment的兩種建立方式

2.Demo示例

(1)示例中左側的按鈕,潘侯爺與碧空海觸發的事件為EventBus的普通事件釋出 (2)左側粘性事件按鈕釋出的為粘性事件

3.實現步驟

本次Demo架構:

3.1導依賴包

使用AndroidStudio2.2。仍然採用在build.gradle下中dependencies下直接新增如下程式碼:

compile 'org.greenrobot:eventbus:3.0.0'

同步後完成依賴新增。

3.2佈局檔案

(1)layout中主佈局檔案,activity_main.xml檔案

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:andro
    xmlns:tools="http://schemas.android.com/tools"
    android:id
="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal" tools:context="com.mly.panhouye.eventbustest.MainActivity"> <LinearLayout android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:orientation="vertical" android:background="#6f6669"> <Button android:layout_gravity="center_horizontal" android:id="@+id/panhouye" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="ŋ" /> <Button android:layout_gravity="center_horizontal" android:id="@+id/bikonghai" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="̿պ" /> <Button android:layout_gravity="center_horizontal" android:id="@+id/postSticky" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="ճДʂ" /> </LinearLayout> <FrameLayout android:id="@+id/framelayout" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="2"></FrameLayout> </LinearLayout>

(2)layout中右側的fragment佈局檔案fragment_msg.xml檔案

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:andro
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView
        android:id="@+id/tv"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:text="no data"
        android:textSize="50sp"
        android:gravity="center_horizontal"/>
</LinearLayout>

(3)layout中粘性事件的演示介面佈局activity_main2.xml檔案

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:andro
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main2"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.mly.panhouye.eventbustest.Main2Activity">
    <TextView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:textSize="30sp"
        android:gravity="center_horizontal"
        android:id="@+id/tv"
        android:text="no data"/>
</RelativeLayout>

3.3java實現程式碼

(1)自定義事件類

本次演示最簡單事件的釋出,事件僅釋出字串資料,MessageEvent.java檔案如下:

package com.mly.panhouye.eventbustest;

/**
 * Created by panchengjia on 2017/2/19 0019.
 */

public class MessageEvent {
    String data;
    public MessageEvent(String data) {
        this.data = data;
    }
}

(2)MsgFragment.java

右側fragment對應的java類,除了在其中關聯其對應的fragment佈局外,還需要新增修改fragment中文字的方法,如下:

package com.mly.panhouye.eventbustest;

import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

/**
 * Created by panchengjia on 2017/2/20 0020.
 */

public class MsgFragment extends Fragment {
    TextView tv;
    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_msg,container,false);
        tv = (TextView) view.findViewById(R.id.tv);
        return view;
    }
    public void setText(String message){
        tv.setText(message);
    }
}

(3)MainActivity.java

MainActivity.java對應的佈局為主佈局,右側的fragment附屬於該佈局,所以需要在該類中註冊EventBus,將當前的Activity註冊為事件訂閱者,具體程式碼如下:

package com.mly.panhouye.eventbustest;

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import org.greenrobot.eventbus.EventBus;
import org.greenrobot.eventbus.Subscribe;
import org.greenrobot.eventbus.ThreadMode;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
    Button panhouye,bikonghai,postSticky;
    MsgFragment msgFragment;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        panhouye= (Button) findViewById(R.id.panhouye);
        bikonghai= (Button) findViewById(R.id.bikonghai);
        postSticky= (Button) findViewById(R.id.postSticky);

        panhouye.setOnClickListener(this);
        bikonghai.setOnClickListener(this);
        postSticky.setOnClickListener(this);
        //新增fragment到右側的幀佈局中
        msgFragment = new MsgFragment();
        getSupportFragmentManager().beginTransaction().add(R.id.framelayout,msgFragment).commit();
    }
    /*個人建議在onResume註冊EventBus
     *在可見可互動狀態下注冊,儘可能少的佔用記憶體
     */
    @Override
    protected void onResume() {
        super.onResume();
        EventBus.getDefault().register(this);
    }
    /*個人建議在onPause註冊EventBus(將當前Activity註冊為事件訂閱者)
     *不影響功能的情況下提早解除註冊,儘可能少的佔用記憶體
     */
    @Override
    protected void onPause() {
        super.onPause();
        EventBus.getDefault().unregister(this);
    }
    /**
     * 事件釋出者(通過按鈕點選事件進行事件釋出)
     * @param v
     */
    @Override
    public void onClick(View v) {
        switch (v.getId()){
            //(1)事件釋出中所傳引數可以作為右側fragment文字的修改內容
            //(2)事件釋出中所傳引數也可以用作事件訂閱者執行方法的區分通知
            case R.id.panhouye:
                EventBus.getDefault().post(new MessageEvent("潘侯爺"));
                break;
            case R.id.bikonghai:
                EventBus.getDefault().post(new MessageEvent("碧空海"));
                break;
            case R.id.postSticky:
                //粘性事件釋出
                EventBus.getDefault().postSticky(new MessageEvent("粘性事件"));
                startActivity(new Intent(this,Main2Activity.class));
                break;
        }
    }
    /**
     * 事件訂閱者自定義的接收方法
     * @param event
     */
    @Subscribe(threadMode = ThreadMode.MAIN)
    public void onMessageEvent(MessageEvent event) {
//        //(1)將事件釋出者釋出的資料作為文字修改內容
//        msgFragment.setText(event.data);
        //(2)將事件釋出者釋出的資料作為方法執行的區分
        switch(event.data){
            case "潘侯爺":
                msgFragment.setText("panhouye");
                break;
            case "碧空海":
                msgFragment.setText("bikonghai");
                break;
        }
    }
}

(4)Main2Activity.java

注意:此佈局作為粘性事件釋出的訂閱者,同樣需要註冊EventBus

package com.mly.panhouye.eventbustest;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;

import org.greenrobot.eventbus.EventBus;
import org.greenrobot.eventbus.Subscribe;
import org.greenrobot.eventbus.ThreadMode;

public class Main2Activity extends AppCompatActivity {
    TextView tv;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main2);
        tv = (TextView) findViewById(R.id.tv);
    }
    @Override
    protected void onResume() {
        super.onResume();
        EventBus.getDefault().register(this);
    }
    @Override
    protected void onPause() {
        super.onPause();
        EventBus.getDefault().unregister(this);
    }
    @Subscribe(threadMode = ThreadMode.MAIN,sticky = true)
    public void onMessageEvent(MessageEvent event) {
//        //(1)將事件釋出者釋出的資料作為文字修改內容
        tv.setText(event.data);
        //(2)將事件釋出者釋出的資料作為方法執行的區分
//        switch(event.data){
//            case "粘性事件":
//                tv.setText("panhouye");
//                break;
//        }
    }
}

釋出的粘性事件在其新訂閱者註冊後將會自動傳遞給新訂閱者,有時我們也需要移除粘性事件,以免它在傳遞下去。

MessageEvent stickyEvent = EventBus.getDefault().getStickyEvent(MessageEvent.class);
// Better check that an event was actually posted before
if(stickyEvent != null) {
      // "Consume" the sticky event
      EventBus.getDefault().removeStickyEvent(stickyEvent);
      // Now do something with it
}
MessageEvent stickyEvent = EventBus.getDefault().removeStickyEvent(MessageEvent.class);
// Better check that an event was actually posted before
if(stickyEvent != null) {
      // Now do something with it
}

4.執行緒模式

EventBus提供了四種執行緒模式:

(1)postThread:使用者將被呼叫在同一個執行緒中,這是釋出事件(這是預設值)。事件傳遞意昧著最少的開銷,因為它完全避免了執行緒切換。因此,這是推薦的模式,來處理簡單的任務,如果是已知的完成是一個很短的時間,而不需要主執行緒。事件處理使用此模式必須迅速返回,以避免阻塞釋出執行緒,這可能是主執行緒。

(2)MainThread:使用者將被呼叫在主執行緒(UI執行緒)。如果釋出執行緒是主執行緒,事件處理程式方法將直接呼叫。使用此模式的事件處理程式必須快速返回,避免阻塞主執行緒。

(3)BackgrounThread:將在後臺執行緒中呼叫訂閱者。如果釋出執行緒不是主執行緒,則事件處理程式方法將被在釋出執行緒中直接呼叫。如果執行緒是主執行緒,eventbus採用單獨的一個後臺執行緒,將按順序呼叫所有的事件。使用此模式的事件處理程式應嘗試快速返回,以避免阻塞後臺執行緒。

(4)Async:事件處理程式方法在一個單獨的執行緒中呼叫。這總是獨立於釋出執行緒和主執行緒。釋出事件從來不會等待使用這種模式的事件處理程式方法。事件處理程式方法使用此模式,如果他們的執行可能需要一段時間,例如用於網路訪問。避免觸發大量在同一時間執行長時間執行的非同步處理程式方法以限制併發執行緒的數目。eventbus使用一個執行緒池來有效地重用已完成的非同步事件處理程式通知的執行緒。