国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 系統(tǒng) > Android > 正文

詳解Android中BroadCastReceiver組件

2020-01-02 07:03:40
字體:
供稿:網(wǎng)友

BroadcastReceiver也就是“廣播接收者”的意思,它是用來接收來自系統(tǒng)和應用中的廣播。

在Android中,Broadcast是一種廣泛運用的在應用程序之間傳輸信息的機制。而BroadcastReceiver是對發(fā)送出來的 Broadcast進行過濾接受并響應的一類組件。

下面將詳細的闡述如何發(fā)送Broadcast和使用BroadcastReceiver過濾接收的過程:

(1)首先在需要發(fā)送信息的地方,把要發(fā)送的信息和用于過濾的信息(如Action、Category)裝入一個Intent對象,然后通過調(diào)用 sendOrderBroadcast()或sendStickyBroadcast()方法,把 Intent對象以廣播方式發(fā)送出去。

(2)當Intent發(fā)送以后,所有已經(jīng)注冊的BroadcastReceiver會檢查注冊時的IntentFilter是否與發(fā)送的Intent相匹配,若匹配則就會調(diào)用BroadcastReceiver的onReceive()方法。所以當我們定義一個BroadcastReceiver的時候,都需要實現(xiàn)onReceive()方法。

注冊BroadcastReceiver有兩種方式

靜態(tài)注冊:在AndroidManifest.xml中用標簽生命注冊,并在標簽內(nèi)用標簽設置過濾器。

<receiver android:name="myRecevice">  //繼承BroadcastReceiver,重寫onReceiver方法    <intent-filter>        <action android:name="com.lc.test"/> //使用過濾器,接收指定action廣播     </intent-filter>  </receiver> 

動態(tài)注冊: 使用IntentFilter在代碼中動態(tài)的注冊一個廣播

 IntentFilter intentFilter = new IntentFilter(); //為BroadcastReceiver指定action,使之用于接收同action的廣播 intentFilter.addAction(String);   registerReceiver(BroadcastReceiver,intentFilter);

 另外值得注意的是,當我們使用動態(tài)注冊時候,當這個Activity或Service被銷毀時如果沒有解除注冊,系統(tǒng)會報一個異常,提示我們是否忘記解除注冊了,所以我們需要在onDestroy()方法中進行解除注冊,一般:在onStart中注冊,onStop中取消unregisterReceiver:

@Override protected void onDestroy() {   super.onDestroy();   unregisterReceiver(receiver); }

  指定廣播目標Action:Intent intent = new Intent(actionString);

  并且可通過Intent攜帶消息 :intent.putExtra(“msg”, “hi,我通過廣播發(fā)送消息了”);

  發(fā)送廣播消息:Context.sendBroadcast(intent )

其中在動態(tài)注冊中可將BroadcastReceiver的繼承類進行封裝,添加構(gòu)造函數(shù)和BroadcastReceiver注冊

BroadcastReceiver的案例演示

首先我們創(chuàng)建一個類MyBroadcastReceiver用于繼承BroadcastReceiver:
這里重寫了一個方法就是接受廣播的意圖并匹配:

//定義的為:public static final String //ACTION_REGISTER_SUCCESS_FINISH="register.success.finish";Constants.ACTION_REGISTER_SUCCESS_FINISH.equals(intent.getAction())

這里是常量:定義如下:

public class MyBroadcastReceiver extends BroadcastReceiver {    @Override    public void onReceive(Context context, Intent intent) {      if (intent != null && Constants.ACTION_REGISTER_SUCCESS_FINISH.equals(intent.getAction())) {        finish();      }    }  }

(1)接下里我們就通過“動態(tài)注冊”的方式使用:

 private MyBroadcastReceiver receiver = new MyBroadcastReceiver();    IntentFilter filter = new IntentFilter();    filter.addAction(BmobConstants.ACTION_REGISTER_SUCCESS_FINISH); //添加action    registerReceiver(receiver, filter); //注冊

(2)靜態(tài)注冊的話是現(xiàn)在清單文件中添加:

<receiver android:name=".MyBroadcastReceiver">       <intent-filter>         <action android:name="register.success.finish"/>         <category android:name="android.intent.category.DEFAULT" />       </intent-filter>     </receiver> 

(2-1)我們就可以使用代碼了:

Intent intent = new Intent("register.success.finish"); intent.putExtra("msg", "hello receiver."); sendBroadcast(intent); //發(fā)送廣播

普通廣播(Normal Broadcast)

普通廣播對于多個接收者來說是完全異步的,通常每個接收者都無需等待即可以接收到廣播,接收者相互之間不會有影響。對于這種廣播,接收者無法終止廣播,即無法阻止其他接收者的接收動作。為了驗證以上論斷,我們新建三個BroadcastReceiver,演示一下這個過程,F(xiàn)irstReceiver、SecondReceiver和ThirdReceiver的代碼如下:

import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.util.Log; public class FirstReceiver extends BroadcastReceiver {   private static final String TAG = "NormalBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = intent.getStringExtra("msg");     Log.i(TAG, "FirstReceiver: " + msg);   } } 
public class SecondReceiver extends BroadcastReceiver {   private static final String TAG = "NormalBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = intent.getStringExtra("msg");     Log.i(TAG, "SecondReceiver: " + msg);   } } 
public class ThirdReceiver extends BroadcastReceiver {   private static final String TAG = "NormalBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = intent.getStringExtra("msg");     Log.i(TAG, "ThirdReceiver: " + msg);   } } 

然后再次點擊發(fā)送按鈕,發(fā)送一條廣播,控制臺打印如下:

看來這三個接收者都接收到這條廣播了,我們稍微修改一下三個接收者,在onReceive方法的最后一行添加以下代碼,試圖終止廣播:

abortBroadcast(); 

再次點擊發(fā)送按鈕,我們會發(fā)現(xiàn),控制臺中三個接收者仍然都打印了自己的日志,表明接收者并不能終止廣播。

有序廣播(Ordered Broadcast)

有序廣播比較特殊,它每次只發(fā)送到優(yōu)先級較高的接收者那里,然后由優(yōu)先級高的接受者再傳播到優(yōu)先級低的接收者那里,優(yōu)先級高的接收者有能力終止這個廣播。
為了演示有序廣播的流程,我們修改一下上面三個接收者的代碼,如下:

import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.os.Bundle; import android.util.Log; public class FirstReceiver extends BroadcastReceiver {   private static final String TAG = "OrderedBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = intent.getStringExtra("msg");     Log.i(TAG, "FirstReceiver: " + msg);     Bundle bundle = new Bundle();     bundle.putString("msg", msg + "@FirstReceiver");     setResultExtras(bundle);   } } 
public class SecondReceiver extends BroadcastReceiver {   private static final String TAG = "OrderedBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = getResultExtras(true).getString("msg");     Log.i(TAG, "SecondReceiver: " + msg);     Bundle bundle = new Bundle();     bundle.putString("msg", msg + "@SecondReceiver");     setResultExtras(bundle);   } }
public class ThirdReceiver extends BroadcastReceiver {   private static final String TAG = "OrderedBroadcast";   @Override   public void onReceive(Context context, Intent intent) {     String msg = getResultExtras(true).getString("msg");     Log.i(TAG, "ThirdReceiver: " + msg);   } } 

我們注意到,在FirstReceiver和SecondReceiver中最后都使用了setResultExtras方法將一個Bundle對象設置為結(jié)果集對象,傳遞到下一個接收者那里,這樣以來,優(yōu)先級低的接收者可以用getResultExtras獲取到最新的經(jīng)過處理的信息集合。
代碼改完之后,我們需要為三個接收者注冊廣播地址,我們修改一下AndroidMainfest.xml文件:

<receiver android:name=".FirstReceiver">   <intent-filter android:priority="1000">     <action android:name="android.intent.action.MY_BROADCAST"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> <receiver android:name=".SecondReceiver">   <intent-filter android:priority="999">     <action android:name="android.intent.action.MY_BROADCAST"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> <receiver android:name=".ThirdReceiver">   <intent-filter android:priority="998">     <action android:name="android.intent.action.MY_BROADCAST"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> 

我們看到,現(xiàn)在這三個接收者的多了一個android:priority屬性,并且依次減小。這個屬性的范圍在-1000到1000,數(shù)值越大,優(yōu)先級越高。
現(xiàn)在,我們需要修改一下發(fā)送廣播的代碼,如下:

public void send(View view) {   Intent intent = new Intent("android.intent.action.MY_BROADCAST");   intent.putExtra("msg", "hello receiver.");   sendOrderedBroadcast(intent, "scott.permission.MY_BROADCAST_PERMISSION"); } 

注意,使用sendOrderedBroadcast方法發(fā)送有序廣播時,需要一個權(quán)限參數(shù),如果為null則表示不要求接收者聲明指定的權(quán)限,如果不為null,則表示接收者若要接收此廣播,需聲明指定權(quán)限。這樣做是從安全角度考慮的,例如系統(tǒng)的短信就是有序廣播的形式,一個應用可能是具有攔截垃圾短信的功能,當短信到來時它可以先接受到短信廣播,必要時終止廣播傳遞,這樣的軟件就必須聲明接收短信的權(quán)限。
所以我們在AndroidMainfest.xml中定義一個權(quán)限:

<permission android:protectionLevel="normal"       android:name="scott.permission.MY_BROADCAST_PERMISSION" /> 

然后聲明使用了此權(quán)限:

<uses-permission android:name="scott.permission.MY_BROADCAST_PERMISSION" /> 

關(guān)于這部分如果有不明白的地方可以參考我之前寫過的一篇文章:Android聲明和使用權(quán)限
然后我們點擊發(fā)送按鈕發(fā)送一條廣播,控制臺打印如下:

我們看到接收是按照順序的,第一個和第二個都在結(jié)果集中加入了自己的標記,并且向優(yōu)先級低的接收者傳遞下去。
既然是順序傳遞,試著終止這種傳遞,看一看效果如何,我們修改FirstReceiver的代碼,在onReceive的最后一行添加以下代碼:

abortBroadcast(); 

然后再次運行程序,控制臺打印如下:

此次,只有第一個接收者執(zhí)行了,其它兩個都沒能執(zhí)行,因為廣播被第一個接收者終止了。
上面就是BroadcastReceiver的介紹,下面我將會舉幾個常見的例子加深一下大家對廣播的理解和應用:

1.開機啟動服務

我們經(jīng)常會有這樣的應用場合,比如消息推送服務,需要實現(xiàn)開機啟動的功能。要實現(xiàn)這個功能,我們就可以訂閱系統(tǒng)“啟動完成”這條廣播,接收到這條廣播后我們就可以啟動自己的服務了。我們來看一下BootCompleteReceiver和MsgPushService的具體實現(xiàn):

import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.util.Log; public class BootCompleteReceiver extends BroadcastReceiver {   private static final String TAG = "BootCompleteReceiver";   @Override   public void onReceive(Context context, Intent intent) {     Intent service = new Intent(context, MsgPushService.class);     context.startService(service);     Log.i(TAG, "Boot Complete. Starting MsgPushService...");   } } 
import android.app.Service; import android.content.Intent; import android.os.IBinder; import android.util.Log; public class MsgPushService extends Service {   private static final String TAG = "MsgPushService";   @Override   public void onCreate() {     super.onCreate();     Log.i(TAG, "onCreate called.");   }   @Override   public int onStartCommand(Intent intent, int flags, int startId) {     Log.i(TAG, "onStartCommand called.");     return super.onStartCommand(intent, flags, startId);   }   @Override   public IBinder onBind(Intent arg0) {     return null;   } } 

然后我們需要在AndroidManifest.xml中配置相關(guān)信息:

<!-- 開機廣播接受者 --> <receiver android:name=".BootCompleteReceiver">   <intent-filter>     <!-- 注冊開機廣播地址-->     <action android:name="android.intent.action.BOOT_COMPLETED"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> <!-- 消息推送服務 --> <service android:name=".MsgPushService"/> 

我們看到BootCompleteReceiver注冊了“android.intent.action.BOOT_COMPLETED”這個開機廣播地址,從安全角度考慮,系統(tǒng)要求必須聲明接收開機啟動廣播的權(quán)限,于是我們再聲明使用下面的權(quán)限:
經(jīng)過上面的幾個步驟之后,我們就完成了開機啟動的功能,將應用運行在模擬器上,然后重啟模擬器,控制臺打印如下:

如果我們查看已運行的服務就會發(fā)現(xiàn),MsgPushService已經(jīng)運行起來了。

2.網(wǎng)絡狀態(tài)變化

在某些場合,比如用戶瀏覽網(wǎng)絡信息時,網(wǎng)絡突然斷開,我們要及時地提醒用戶網(wǎng)絡已斷開。要實現(xiàn)這個功能,我們可以接收網(wǎng)絡狀態(tài)改變這樣一條廣播,當由連接狀態(tài)變?yōu)閿嚅_狀態(tài)時,系統(tǒng)就會發(fā)送一條廣播,我們接收到之后,再通過網(wǎng)絡的狀態(tài)做出相應的操作。下面就來實現(xiàn)一下這個功能:

import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.net.ConnectivityManager; import android.net.NetworkInfo; import android.util.Log; import android.widget.Toast; public class NetworkStateReceiver extends BroadcastReceiver {   private static final String TAG = "NetworkStateReceiver";   @Override   public void onReceive(Context context, Intent intent) {     Log.i(TAG, "network state changed.");     if (!isNetworkAvailable(context)) {       Toast.makeText(context, "network disconnected!", 0).show();     }   }   /**    * 網(wǎng)絡是否可用    */   public static boolean isNetworkAvailable(Context context) {     ConnectivityManager mgr = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);     NetworkInfo[] info = mgr.getAllNetworkInfo();     if (info != null) {       for (int i = 0; i < info.length; i++) {         if (info[i].getState() == NetworkInfo.State.CONNECTED) {           return true;         }       }     }     return false;   } } 

再注冊一下這個接收者的信息:

<receiver android:name=".NetworkStateReceiver">   <intent-filter>     <action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> 

因為在isNetworkAvailable方法中我們使用到了網(wǎng)絡狀態(tài)相關(guān)的API,所以需要聲明相關(guān)的權(quán)限才行,下面就是對應的權(quán)限聲明:

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> 

我們可以測試一下,比如關(guān)閉WiFi,看看有什么效果。

3.電量變化

如果我們閱讀軟件,可能是全屏閱讀,這個時候用戶就看不到剩余的電量,我們就可以為他們提供電量的信息。要想做到這一點,我們需要接收一條電量變化的廣播,然后獲取百分比信息,這聽上去挺簡單的,我們就來實現(xiàn)以下:

import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.os.BatteryManager; import android.util.Log; public class BatteryChangedReceiver extends BroadcastReceiver {   private static final String TAG = "BatteryChangedReceiver";   @Override   public void onReceive(Context context, Intent intent) {     int currLevel = intent.getIntExtra(BatteryManager.EXTRA_LEVEL, 0); //當前電量     int total = intent.getIntExtra(BatteryManager.EXTRA_SCALE, 1);   //總電量     int percent = currLevel * 100 / total;     Log.i(TAG, "battery: " + percent + "%");   } } 

然后再注冊一下廣播接地址信息就可以了:

<receiver android:name=".BatteryChangedReceiver">   <intent-filter>     <action android:name="android.intent.action.BATTERY_CHANGED"/>     <category android:name="android.intent.category.DEFAULT" />   </intent-filter> </receiver> 

當然,有些時候我們是要立即獲取電量的,而不是等電量變化的廣播,比如當閱讀軟件打開時立即顯示出電池電量。我們可以按以下方式獲取:

Intent batteryIntent = getApplicationContext().registerReceiver(null,     new IntentFilter(Intent.ACTION_BATTERY_CHANGED)); int currLevel = batteryIntent.getIntExtra(BatteryManager.EXTRA_LEVEL, 0); int total = batteryIntent.getIntExtra(BatteryManager.EXTRA_SCALE, 1); int percent = currLevel * 100 / total; Log.i("battery", "battery: " + percent + "%"); 

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
夜夜嗨yeyeh| 免费a级毛片在线播放| 黄网站在线观看高清免费| wwww在线观看| 日本视频一二三区中文字幕| 免费a级人成a大片在线观看| 男女午夜视频在线观看| 激情丁香婷婷| 国产福利图片| 免费女人毛片视频| 国产精品视频一区二区免费不卡| 国产黄色片在线观看| 在线观看av资源网| av在线播放av| 1区2区3区在线| 日本高清中文字幕二区在线| 国产一级二级三级在线观看| 国产国语**毛片高清视频 | 天堂资源在线中文| 国产成人精品男人的天堂538| www.狠狠插| 国产二区视频在线观看| 国内外激情在线| a视频免费看| 69av二区| 国产三级视频在线看| 夜夜操com| 136福利第一导航国产在线| 久蕉依人在线视频| 羞羞视频在线免费看| 9999在线视频| 国产第一页在线| 国产一级二级三级在线观看| 黄色三级视频在线观看| а√天堂www在线а√天堂视频| 国产视频三区| 国产精品日日爱| 91福利在线免费| 99视频资源网| 天堂中文字幕在线| 国产精选在线观看| 另类综合图区| 免费在线超碰| 免费观看久久久久| 噜噜噜噜噜在线视频| 天堂网中文在线| 午夜在线视频| baoyu777.永久免费视频| www.操操操| 久久er视频| 国产一起色一起爱| 国产区av在线| 精品视频一区二区观看| 亚洲综合在线网| 欧美成人久久电影香蕉| 国产免费麻豆视频| 九九免费视频| 国产区视频在线| 精品久久av| 麻豆福利在线观看| 国产一区二区三区不卡免费观看 | 欧美高清xxxx性| 久草亚洲一区| 国产欧美一区二区三区小说| 精品电影在线| 免费在线播放av| 国产系列在线观看| 日本视频一二三区中文字幕| 狠狠操狠狠色| 国产高清一级片| 久热av在线| 91高清国产| 最近中文字幕av免费高清| 精品国产免费第一区二区| 蜜桃av网站| 精品推荐国产麻豆剧传媒| 交视频在线观看国产| 国产福利电影在线观看| 九九热视频免费观看| 91中文字幕网| 最近中文av字幕在线中文| 日本不卡影院| 在线免费观看黄色片| 最近中文字幕mv2018在线高清| 国产a级网站| 丁香视频免费观看| 欧美视频免费一区二区三区| 依依成人在线| 国产超碰97| 国产经典av| 国产一卡2卡3卡免费网站| 狠狠操视频网| 欧美亚洲天堂| 免费网站看黄yyy222| 精品成人免费自拍视频| 国产精品黄页网站在线播放免费| 成视人a免费观看视频 | free性亚洲| 精品国产二区三区| 国产秀色在线www免费观看| 黄色片免费在线| 久久久久久国产视频| 国产蜜臀av在线播放| 成人亚洲一区二区三区| 丁香花视频在线观看| 中文字幕人成高视频| 国产三级视频| 99久久国产视频| www.综合网.com| 国内外激情在线| 国产一区二区三区不卡免费观看| 午夜视频免费在线观看| 国产网红在线观看| 6699久久国产精品免费| 精品国产免费第一区二区| 亚洲人成电影| 国产中文字幕在线观看| 中文字幕麻豆| 中文字幕av在线播放| www.色婷婷| 国产美女高潮| 91在线超碰| 亚洲激情丁香| 青青草视频在线观看| h网站免费在线观看| 超碰在线中文| 天天干天天摸| 天天操夜夜添| 中文字幕在线播放网址| 国产天堂在线观看| 在线午夜视频| 麻豆精品永久免费视频| 国产视频三区| jlzzjlzz欧美大全| 国产精品国产国产aⅴ| 亚洲91av| 国产黄视频在线观看| 四虎成人欧美精品在永久在线 | 在线一二三区| 狠狠操视频网| 亚洲精品视频在线免费| 丁香视频免费观看| av亚洲在线| 18加网站在线| 全网国产福利在线播放| 99爱视频在线观看| 国产精品视频h| 尤物视频免费在线观看| 快射av在线播放一区| wwww亚洲| 天天操夜夜做| 在线国产网址| 在线视频二区| 精品极品三级久久久久| 国产成人精品综合网站| 国产激情在线观看| 国产黄a三级三级三级av在线看| 天堂中文字幕在线| 精品中文字幕不卡在线视频| 一区二区三区四区在线免费视频| 国产网站麻豆精品视频| 中文字幕在线视频观看| 国产男女无套在线播放| 成在在线免费视频| 国产综合视频一区二区三区免费| 欧美色第一页| 日本免费不卡| 久久久久久久久久久久网站| 欧美日韩在线中文字幕| 久色视频在线观看| 国产在线播放av| 国产色在线观看| 国产福利微拍精品一区二区| 国产视频2区| 99高清免费国产自产拍| 亚洲va国产日韩欧美精品色婷婷| 在线播放黄色网址| 麻豆av在线| 国产黄色免费在线观看| 99免费视频| 二区三区中文字幕| 国产特级毛片| 亚洲成人福利| 国产女人在线观看| 国产成人精品男人的天堂538| 伊人精品影院| 国产九色porn网址| 国产高清免费在线播放| av免费在线观看网站| 免费看的毛片| 97视频网站| 国产秒拍福利视频露脸| 中文字幕av在线| 高清av中文在线字幕观看1| 国产高潮av| 国产精品㊣新片速递bt| 亚洲伊人网在线观看| 中文字幕在线观看播放| 亚洲成人av高清| 国产精品伦一区二区三区视频|