編輯:關於Android編程
OTTO是一個EventBus類型的事件傳輸總線,它可以提供“存儲轉發”的功能,讓你APP中各個組件的交流更加便利,讓你的程序分層更加清晰。
使用場景
OTTO基於Observer設計模式。它有發布者,訂閱者這兩個主要對象。OTTO的最佳實踐就是通過反射犧牲了微小的性能,同時極大的提高了程序的耦合度,更加利於MVP分工開發與維護。業務層開發者在處理資源(比如Db, REST等)後並發布消息,展示層開發者(比如Activity/Fragment)就可以處理消息,而不用關心數據是怎麼來的(在讀報紙的時候需要知道編輯們如何排版印刷嗎?),比如:
Fragment,Service或者Activity組件之間的通信。比如
導航菜單的NavigationDrawer與Activity的通信
Activity與Activity的通信(在設置界面上勾選了夜間模式,回到主界面就發現已經完成變色了;或者你在詳細界面上點了一個贊,回到主界面發現已經同步增加了一個"贊")
MVP(Model View Presidenter)架構中,Model與Presidenter的回掉通信。包括但不限於REST, DB, SP, BroadcastReceiver, ContentObserver。

一、Android Studio中配置Otto (Eclipse中直接下載jar包導入)
跟之前介紹的其他的框架一樣,它只需要簡單地在build.gradle中配置下面的部分即可
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
compile 'com.android.support:appcompat-v7:19.+'
/
/otto 所需要依賴的包
}
二、開始使用
1. 訂閱者
當你想進行訂閱時,首先要打開注冊(比如生命周期的啟動/恢復部分)
Bus.register(this);
當你不再關注某個事情後,需要取消注冊(比如生命周期的停止/暫停部分)
Bus.unregister(this);
當你對某個事情感興趣的話,就加入@Subscribe進行關注。
@Subscribe public void getMessage(@NonNull SomeEvent s) {
//TODO: 在回掉中使用這個事件
}
2. 發布者
當你想發布消息時,使用post即可,注意發布者同樣需要事先注冊。
Bus.post(SomeEvent);
實例說明
本文以短信服務接受為例,介紹OTTO的使用
1. 構造單例模式
BUS是一個單例,所以我們要創建一個單例模式,而最簡單的單例當然是在Application中建立,記得在Manifest注冊哦。
/**
* Created by leon on 15/5/27.
* 主線程事件總線,方便在異步任務中回掉
*/
public class MainThreadBus extends Bus {
private final Handler handler = new Handler(Looper.getMainLooper());
@Override public void post(final Object event) {
if (Looper.myLooper() == Looper.getMainLooper()) {
//直接通過反射調用
super.post(event);
} else {
//通過handler把異步任務發送到UI線程,然後再反射調用
handler.post(new Runnable() {
@Override
public void run() {
MainThreadBus.super.post(event);
}
});
}
}
}
接著是Application的重寫
public class GlobalContext extends Application {
//event bus singleton
public static final MainThreadBus bus = new MainThreadBus();
public static GlobalContext instance;
@Override public void onCreate() {
super.onCreate();
instance = this;
}
public static MainThreadBus getBusInstance(){
return bus;
}
public static GlobalContext getContextInstance(){
return instance;
}
}
2. 發送者(Publisher)
現在我們要注冊一個短信接收機器,為了與國內毒瘤軟件搶占第一的短信監聽,我們使用Service動態注冊接收器,以搶占第一優先級。
建立一個常駐後台的服務,動態注冊接收器,以搶占第一優先級。同樣記得要把Service在Manifest注冊哦。
public class SmsService extends Service {
private SmsReceiver mReceiver = null;
public SmsService() {
}
@Override public IBinder onBind(Intent intent) {
// TODO: Return the communication channel to the service.
throw new UnsupportedOperationException("Not yet implemented");
}
@Override public void onCreate() {
super.onCreate();
Log.d(TAG, "onCreate");
IntentFilter iFilter = null; // 意圖過濾對象
mReceiver = new SmsReceiver(); // 廣播接收類初始化
iFilter = new IntentFilter("android.provider.Telephony.SMS_RECEIVED");
iFilter.setPriority(Integer.MAX_VALUE); // 設置優先級
GlobalContext.getBusInstance().register(mReceiver);//注冊Bus
registerReceiver(mReceiver, iFilter); // 注冊廣播
}
@Override public void onDestroy() {
super.onDestroy();
if (mReceiver != null){
GlobalContext.getBusInstance().unregister(mReceiver);//取消注冊Bus
unregisterReceiver(mReceiver);
}
}
}
接下來就是真正的發布者SmsReceiver了,大部分代碼與網上都是一樣的,注意看我是如何發布消息的。
public class SmsReceiver extends BroadcastReceiver {
public SmsReceiver() {
}
@Override public void onReceive(Context context, Intent intent) {
Bundle bundle = intent.getExtras();
//獲取鏈路層的協議數據單元
Object[] messages = (Object[]) bundle.get("pdus");
SmsMessage[] sms = new SmsMessage[messages.length];
// Create messages for each incoming PDU
for (int n = 0; n < messages.length; n++) {
sms[n] = SmsMessage.createFromPdu((byte[]) messages[n]);
}
for (SmsMessage msg : sms) {
//TODO: 這裡應該加上你自己的過濾條件,比如手機號,短信內容
//盡可能的攔截短信,這個命令在MIUI,flyme上都沒有用
abortBroadcast();
GlobalContext.getBusInstance().post(msg);
}
}
}
從代碼中可以看出,在Service中,我們控制了SmsReceiver中BUS的生命周期,之後在SmsReceiver中,通過post把消息發布出去
GlobalContext.getBusInstance().post(msg);
3. 接收者(Subscriber)
接收者可以是Activity,也可以是Fragment,還可以是Service。
我們以Fragment為例進行操作.
public class SMSControlFragment extends Fragment {
Bus bus = GlobalContext.getBusInstance();
@Override public void onAttach(Activity activity) {
super.onAttach(activity);
bus.register(this);
}
@Override public void onDetach() {
super.onDetach();
bus.unregister(this);
}
@Subscribe public void getMessage(SmsMessage s) {
mTvNumber.setText(s.getOriginatingAddress());
mTvMessage.setText(s.getMessageBody());
}
}
訂閱者是Fragment,發布者是SmsReceiver,消息內容是SmsMessage。
通過以上操作,一個使用OTTO的消息傳遞總線就完成了。
4.綜合demo
下面的Demo, 僅為了讓大家知道“事件”被產生了之後,post出來,所有訂閱了該事件的類都會接到該事件,接受的先後順序,不由我們控制!
public class MyActivity extends ActionBarActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_my);
findViewById(R.id.button_change).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
BusProvider.getBusInstance().post(new DataChangedEvent("this is changed String"));//發布事件
}
});
}
@Override
protected void onResume() {
super.onResume();
BusProvider.getBusInstance().register(this);//注冊
}
@Override
protected void onPause() {
super.onPause();
BusProvider.getBusInstance().unregister(this);//注銷
}
@Subscribe //訂閱事件DataChangedEvent
public void sayGoodOnEvent(DataChangedEvent event){
Log.e("event", "good");
}
@Subscribe //訂閱事件
public void sayBadOnEvent(DataChangedEvent event){
Log.e("event", "bad");
}
@Produce //產生事件
public DataChangedEvent produceDataChangedEvent(){
return new DataChangedEvent("this is changed String");
}
}
android之存儲數據的四大方式
下面詳細解釋這四大方式的特點第一種:文件存儲數據核心原理: Context提供了兩個方法來打開數據文件裡的文件IO流 FileInputStream openFileIn
我的Android進階之旅------)Android編譯錯誤java.util.zip.ZipException: duplicate entry的解決方法
今天在Android Studio中把另外一個項目引入當前項目,編譯的時候出現了java.util.zip.ZipException: duplicate entry錯誤
Android 下拉刷新上拉加載效果功能,使用開源項目android-pulltorefresh實現
應用場景: 在App開發中,對於信息的獲取與演示,不可能全部將其獲取與演示,為了在用戶使用中,給予用戶以友好、方便的用戶體驗,以滑動、下拉的效果動態加載數據的要求就會出
關於Android事件派發流程的理解
以前看了很多人介紹的Android事件派發流程,但最近使用那些來寫代碼的時候出現了不少錯誤。所以回顧一下整個流程,簡單介紹從手觸摸屏幕開始到事件在View樹派發。從源碼上