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

来源:http://www.cnblogs.com/panhouye/archive/2017/02/20/6420727.html
-Advertisement-
Play Games

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


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:android="http://schemas.android.com/apk/res/android"
    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:android="http://schemas.android.com/apk/res/android"
    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:android="http://schemas.android.com/apk/res/android"
    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使用一個線程池來有效地重用已完成的非同步事件處理程式通知的線程。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一.Css規則主要由兩部分組成 1.選擇器 2.一條或多條聲明a. 選擇器主要作用是為了確定需要改變樣式的HTML元素b. 每一條聲明由一個屬性和一個值組成,使用花括弧來包圍聲明,屬性與值之間使用冒號(:)分開,多條聲明用分號(;)隔開. 例如:form {width: 200px;margin: ...
  • 1、Math.trunc() 該方法用於取出一個小數的小數部分,返回整數部分。看例子吧: 2、Math.sign() 該方法用來判斷一個數到底是正數,負數,還是0。有五中返回值。看下麵例子吧: 參數為正數時,返回1; 參數為0時,返回0; 參數為-0時,返回-0; 參數為負數時,返回-1; 參數為其 ...
  • 頁面上使用js寫了一個獲取後臺數據的方法 }); tab.html(parentStr); } }) } 其中的 發現點擊無效無效 原來是 ajax載入新dom之前js 就載入完了,事件當然沒有綁定到新載入的dom上 解決方法: 使用jquery的委托事件,將該方法委托到頁面已經存在的一個節點上 問 ...
  • 由於新項目的的需求,需要畫環形圖,由於以前都沒接觸過這一類(我是菜鳥),去cocochina山找到了一個案例,個人覺得還可以,分享一下 github 地址https://github.com/zhouxing5311/ZZCircleProgress 這個使用起來非常的簡單 作者開放了很多介面,可以 ...
  • 導語:之前做習慣了Framework層的開發,今天在武漢鬥魚公司面試APP客戶端的開發,其中一道題是講述Asynctask的優缺點,我靠,我只是知道有這麼一個東西,會用而已,看來之前的生活太過於安逸,這麼基礎的東西都不會,特意回家總結下。 轉載請註明原文鏈接:http://www.cnblogs.c ...
  • 寫項目的時候就遇到了這個尷尬的問題,有的是需要進行網路請求的,或者有的手機有點卡頓,用戶就會連續點擊,經常出現的就是接連Push兩個同樣的控制器,當然還有重覆調用網路請求的方法,等等等等。。。。 比較笨的方法就是在每個點擊的地方把用戶交互給關掉,等到功能實現後之後再將這個按鈕的用戶交互恢復。 當然了 ...
  • 很少用DatePicker,項目需要用到DatePicker,再用時發現跟以前有些變化,重新簡單梳理一下DatePicker的用法。 先上效果圖: 首先是xml文件的用法: 以前是用android:spinnersShown和android:calendarViewShown這兩個屬性來控制Date ...
  • 現在做的項目遇到一個問題,設計給過來的圖片只有一種狀態,但是實現的需求是要求有兩個狀態,另一種選狀態為圖片背景加邊框。如圖: 剛開使用使用ImageView ,ImageViewButton 效果不是很明顯; 後來發現 layer-list 能很好的實現這個效果,先分別建 正常模式與選中模式的xml ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...