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之間的交互(兩種實現方式)
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使用一個線程池來有效地重用已完成的非同步事件處理程式通知的線程。