上節(jié)我們對(duì)BroadcastReceiver已經(jīng)有了一個(gè)初步的了解了窿侈,知道兩種廣播類型:標(biāo)準(zhǔn)與有序磷醋, 動(dòng)態(tài)或靜態(tài)注冊(cè)廣播接收者冕臭,監(jiān)聽系統(tǒng)廣播,自己發(fā)送廣播卷胯!已經(jīng)滿足我們的基本需求了~ 但是前面寫的廣播都是全局廣播子刮!這同樣意味著我們APP發(fā)出的廣播,其他APP都會(huì)接收到窑睁, 或者其他APP發(fā)送的廣播挺峡,我們的APP也同樣會(huì)接收到,這樣容易引起一些安全性的問題担钮!而 Android中給我們提供了本地廣播的機(jī)制橱赠,使用該機(jī)制發(fā)出的廣播只會(huì)在APP內(nèi)部傳播,而且 廣播接收者也只能收到本應(yīng)用發(fā)出的廣播箫津!
1.本地廣播
1)核心用法:
PS:本地廣播無(wú)法通過靜態(tài)注冊(cè)方式來(lái)接受,相比起系統(tǒng)全局廣播更加高效
2)注意事項(xiàng):
3)代碼示例(別處登陸踢用戶下線):
像微信一樣狭姨,正在運(yùn)行的微信,如果我們用別的手機(jī)再次登陸自己的賬號(hào)苏遥,前面這個(gè)是會(huì)提醒賬戶 在別的終端登錄這樣饼拍,然后把我們打開的所有Activity都關(guān)掉,然后回到登陸頁(yè)面這樣~
下面我們就來(lái)寫個(gè)簡(jiǎn)單的例子:
運(yùn)行效果圖:
代碼實(shí)現(xiàn):
Step 1:準(zhǔn)備一個(gè)關(guān)閉所有Activity的ActivityCollector 田炭,這里之前用前面Activity提供的那個(gè)!
ActivityCollector.java
public class ActivityCollector {
private static List<Activity> activities = new ArrayList<Activity>();
public static void addActivity(Activity activity) {
activities.add(activity);
}
public static void removeActivity(Activity activity) {
activities.remove(activity);
}
public static void finishAll() {
for (Activity activity : activities) {
if (!activity.isFinishing()) {
activity.finish();
}
}
}
}
Step 2:先寫要給簡(jiǎn)單的BaseActivity惕耕,用來(lái)繼承,接著寫下登陸界面诫肠!
public class BaseActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityCollector.addActivity(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
ActivityCollector.removeActivity(this);
}
}
LoginActivity.java:
public class LoginActivity extends BaseActivity implements View.OnClickListener{
private SharedPreferences pref;
private SharedPreferences.Editor editor;
private EditText edit_user;
private EditText edit_pawd;
private Button btn_login;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
pref = PreferenceManager.getDefaultSharedPreferences(this);
bindViews();
}
private void bindViews() {
edit_user = (EditText) findViewById(R.id.edit_user);
edit_pawd = (EditText) findViewById(R.id.edit_pawd);
btn_login = (Button) findViewById(R.id.btn_login);
btn_login.setOnClickListener(this);
}
@Override
protected void onStart() {
super.onStart();
if(!pref.getString("user","").equals("")){
edit_user.setText(pref.getString("user",""));
edit_pawd.setText(pref.getString("pawd",""));
}
}
@Override
public void onClick(View v) {
String user = edit_user.getText().toString();
String pawd = edit_pawd.getText().toString();
if(user.equals("123")&&pawd.equals("123")){
editor = pref.edit();
editor.putString("user", user);
editor.putString("pawd", pawd);
editor.commit();
Intent intent = new Intent(LoginActivity.this, MainActivity.class);
startActivity(intent);
Toast.makeText(LoginActivity.this,"喲,竟然蒙對(duì)了~",Toast.LENGTH_SHORT).show();
finish();
}else{
Toast.makeText(LoginActivity.this,"這么簡(jiǎn)單都輸出欺缘,腦子呢栋豫?",Toast.LENGTH_SHORT).show();
}
}
}
Step 3:自定義一個(gè)BroadcastReceiver,在onReceive里完成彈出對(duì)話框操作谚殊,以及啟動(dòng)登陸頁(yè)面:
MyBcReceiver.java
public class MyBcReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context, Intent intent) {
AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
dialogBuilder.setTitle("警告:");
dialogBuilder.setMessage("您的賬號(hào)在別處登錄丧鸯,請(qǐng)重新登陸~");
dialogBuilder.setCancelable(false);
dialogBuilder.setPositiveButton("確定",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
ActivityCollector.finishAll();
Intent intent = new Intent(context, LoginActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
});
AlertDialog alertDialog = dialogBuilder.create();
alertDialog.getWindow().setType(
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
alertDialog.show();
}
}
別忘了AndroidManifest.xml中加上系統(tǒng)對(duì)話框權(quán)限: < uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
Step 4:在MainActivity中,實(shí)例化localBroadcastManager嫩絮,拿他完成相關(guān)操作丛肢,另外銷毀時(shí) 注意unregisterReceiver!
MainActivity.java
public class MainActivity extends BaseActivity {
private MyBcReceiver localReceiver;
private LocalBroadcastManager localBroadcastManager;
private IntentFilter intentFilter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
localBroadcastManager = LocalBroadcastManager.getInstance(this);
//初始化廣播接收者剿干,設(shè)置過濾器
localReceiver = new MyBcReceiver();
intentFilter = new IntentFilter();
intentFilter.addAction("com.jay.mybcreceiver.LOGIN_OTHER");
localBroadcastManager.registerReceiver(localReceiver, intentFilter);
Button btn_send = (Button) findViewById(R.id.btn_send);
btn_send.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.jay.mybcreceiver.LOGIN_OTHER");
localBroadcastManager.sendBroadcast(intent);
}
});
}
@Override
protected void onDestroy() {
super.onDestroy();
localBroadcastManager.unregisterReceiver(localReceiver);
}
}
好的蜂怎,就是這么簡(jiǎn)單,別忘記注冊(cè)Activity哦~
2.Android 4.3以上版本監(jiān)聽開機(jī)啟動(dòng)廣播的問題解決:
在Android 4.3以上的版本置尔,允許我們將應(yīng)用安裝在SD上杠步,我們都知道是系統(tǒng)開機(jī) 間隔一小段時(shí)間后,才裝載SD卡的,這樣我們的應(yīng)用就可能監(jiān)聽不到這個(gè)廣播了幽歼! 所以我們需要既監(jiān)聽開機(jī)廣播又監(jiān)聽SD卡掛載廣播朵锣!
另外,有些手機(jī)可能并沒有SD卡甸私,所以這兩個(gè)廣播監(jiān)聽我們不能寫到同一個(gè)Intetn-filter里 而是應(yīng)該寫成兩個(gè)诚些,配置代碼如下:
<receiver android:name=".MyBroadcastReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
<intent-filter>
<action android:name="ANDROID.INTENT.ACTION.MEDIA_MOUNTED"/>
<action android:name="ANDROID.INTENT.ACTION.MEDIA_UNMOUNTED"/>
<data android:scheme="file"/>
</intent-filter>
</receiver>
3.常用的系統(tǒng)廣播總結(jié):
最后給大家提供下我們平常可能會(huì)用到的一些系統(tǒng)廣播吧:
Intent.ACTION_AIRPLANE_MODE_CHANGED;
//關(guān)閉或打開飛行模式時(shí)的廣播
<strong>Intent.ACTION_BATTERY_CHANGED;
//充電狀態(tài)皇型,或者電池的電量發(fā)生變化
//電池的充電狀態(tài)诬烹、電荷級(jí)別改變,不能通過組建聲明接收這個(gè)廣播犀被,只有通過Context.registerReceiver()注冊(cè)
<strong>Intent.ACTION_BATTERY_LOW;
//表示電池電量低
<strong>Intent.ACTION_BATTERY_OKAY;
//表示電池電量充足椅您,即從電池電量低變化到飽滿時(shí)會(huì)發(fā)出廣播
Intent.ACTION_BOOT_COMPLETED;
//在系統(tǒng)啟動(dòng)完成后,這個(gè)動(dòng)作被廣播一次(只有一次)寡键。
Intent.ACTION_CAMERA_BUTTON;
//按下照相時(shí)的拍照按鍵(硬件按鍵)時(shí)發(fā)出的廣播
Intent.ACTION_CLOSE_SYSTEM_DIALOGS;
//當(dāng)屏幕超時(shí)進(jìn)行鎖屏?xí)r,當(dāng)用戶按下電源按鈕,長(zhǎng)按或短按(不管有沒跳出話框)掀泳,進(jìn)行鎖屏?xí)r,android系統(tǒng)都會(huì)廣播此Action消息
Intent.ACTION_CONFIGURATION_CHANGED;
//設(shè)備當(dāng)前設(shè)置被改變時(shí)發(fā)出的廣播(包括的改變:界面語(yǔ)言,設(shè)備方向西轩,等员舵,請(qǐng)參考Configuration.java)
Intent.ACTION_DATE_CHANGED;
//設(shè)備日期發(fā)生改變時(shí)會(huì)發(fā)出此廣播
Intent.ACTION_DEVICE_STORAGE_LOW;
//設(shè)備內(nèi)存不足時(shí)發(fā)出的廣播,此廣播只能由系統(tǒng)使用,其它APP不可用藕畔?
Intent.ACTION_DEVICE_STORAGE_OK;
//設(shè)備內(nèi)存從不足到充足時(shí)發(fā)出的廣播,此廣播只能由系統(tǒng)使用马僻,其它APP不可用?
Intent.ACTION_DOCK_EVENT;
//
//發(fā)出此廣播的地方frameworks\base\services\java\com\android\server\DockObserver.java
Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;
////移動(dòng)APP完成之后注服,發(fā)出的廣播(移動(dòng)是指:APP2SD)
Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;
//正在移動(dòng)APP時(shí)韭邓,發(fā)出的廣播(移動(dòng)是指:APP2SD)
Intent.ACTION_GTALK_SERVICE_CONNECTED;
//Gtalk已建立連接時(shí)發(fā)出的廣播
Intent.ACTION_GTALK_SERVICE_DISCONNECTED;
//Gtalk已斷開連接時(shí)發(fā)出的廣播
Intent.ACTION_HEADSET_PLUG;
//在耳機(jī)口上插入耳機(jī)時(shí)發(fā)出的廣播
Intent.ACTION_INPUT_METHOD_CHANGED;
//改變輸入法時(shí)發(fā)出的廣播
Intent.ACTION_LOCALE_CHANGED;
//設(shè)備當(dāng)前區(qū)域設(shè)置已更改時(shí)發(fā)出的廣播
Intent.ACTION_MANAGE_PACKAGE_STORAGE;
//
Intent.ACTION_MEDIA_BAD_REMOVAL;
//未正確移除SD卡(正確移除SD卡的方法:設(shè)置--SD卡和設(shè)備內(nèi)存--卸載SD卡),但已把SD卡取出來(lái)時(shí)發(fā)出的廣播
//廣播:擴(kuò)展介質(zhì)(擴(kuò)展卡)已經(jīng)從 SD 卡插槽拔出溶弟,但是掛載點(diǎn) (mount point) 還沒解除 (unmount)
Intent.ACTION_MEDIA_BUTTON;
//按下"Media Button" 按鍵時(shí)發(fā)出的廣播,假如有"Media Button" 按鍵的話(硬件按鍵)
Intent.ACTION_MEDIA_CHECKING;
//插入外部?jī)?chǔ)存裝置女淑,比如SD卡時(shí),系統(tǒng)會(huì)檢驗(yàn)SD卡辜御,此時(shí)發(fā)出的廣播?
Intent.ACTION_MEDIA_EJECT;
//已拔掉外部大容量?jī)?chǔ)存設(shè)備發(fā)出的廣播(比如SD卡鸭你,或移動(dòng)硬盤),不管有沒有正確卸載都會(huì)發(fā)出此廣播?
//廣播:用戶想要移除擴(kuò)展介質(zhì)(拔掉擴(kuò)展卡)。
Intent.ACTION_MEDIA_MOUNTED;
//插入SD卡并且已正確安裝(識(shí)別)時(shí)發(fā)出的廣播
//廣播:擴(kuò)展介質(zhì)被插入擒权,而且已經(jīng)被掛載袱巨。
Intent.ACTION_MEDIA_NOFS;
//
Intent.ACTION_MEDIA_REMOVED;
//外部?jī)?chǔ)存設(shè)備已被移除,不管有沒正確卸載,都會(huì)發(fā)出此廣播碳抄?
// 廣播:擴(kuò)展介質(zhì)被移除愉老。
Intent.ACTION_MEDIA_SCANNER_FINISHED;
//廣播:已經(jīng)掃描完介質(zhì)的一個(gè)目錄
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
//
Intent.ACTION_MEDIA_SCANNER_STARTED;
//廣播:開始掃描介質(zhì)的一個(gè)目錄
Intent.ACTION_MEDIA_SHARED;
// 廣播:擴(kuò)展介質(zhì)的掛載被解除 (unmount),因?yàn)樗呀?jīng)作為 USB 大容量存儲(chǔ)被共享纳鼎。
Intent.ACTION_MEDIA_UNMOUNTABLE;
//
Intent.ACTION_MEDIA_UNMOUNTED
// 廣播:擴(kuò)展介質(zhì)存在俺夕,但是還沒有被掛載 (mount)裳凸。
Intent.ACTION_NEW_OUTGOING_CALL;
Intent.ACTION_PACKAGE_ADDED;
//成功的安裝APK之后
//廣播:設(shè)備上新安裝了一個(gè)應(yīng)用程序包。
//一個(gè)新應(yīng)用包已經(jīng)安裝在設(shè)備上劝贸,數(shù)據(jù)包括包名(最新安裝的包程序不能接收到這個(gè)廣播)
Intent.ACTION_PACKAGE_CHANGED;
//一個(gè)已存在的應(yīng)用程序包已經(jīng)改變姨谷,包括包名
Intent.ACTION_PACKAGE_DATA_CLEARED;
//清除一個(gè)應(yīng)用程序的數(shù)據(jù)時(shí)發(fā)出的廣播(在設(shè)置--應(yīng)用管理--選中某個(gè)應(yīng)用,之后點(diǎn)清除數(shù)據(jù)時(shí)?)
//用戶已經(jīng)清除一個(gè)包的數(shù)據(jù)映九,包括包名(清除包程序不能接收到這個(gè)廣播)
Intent.ACTION_PACKAGE_INSTALL;
//觸發(fā)一個(gè)下載并且完成安裝時(shí)發(fā)出的廣播梦湘,比如在電子市場(chǎng)里下載應(yīng)用?
//
Intent.ACTION_PACKAGE_REMOVED;
//成功的刪除某個(gè)APK之后發(fā)出的廣播
//一個(gè)已存在的應(yīng)用程序包已經(jīng)從設(shè)備上移除件甥,包括包名(正在被安裝的包程序不能接收到這個(gè)廣播)
Intent.ACTION_PACKAGE_REPLACED;
//替換一個(gè)現(xiàn)有的安裝包時(shí)發(fā)出的廣播(不管現(xiàn)在安裝的APP比之前的新還是舊捌议,都會(huì)發(fā)出此廣播?)
Intent.ACTION_PACKAGE_RESTARTED;
//用戶重新開始一個(gè)包引有,包的所有進(jìn)程將被殺死瓣颅,所有與其聯(lián)系的運(yùn)行時(shí)間狀態(tài)應(yīng)該被移除,包括包名(重新開始包程序不能接收到這個(gè)廣播)
Intent.ACTION_POWER_CONNECTED;
//插上外部電源時(shí)發(fā)出的廣播
Intent.ACTION_POWER_DISCONNECTED;
//已斷開外部電源連接時(shí)發(fā)出的廣播
Intent.ACTION_PROVIDER_CHANGED;
//
Intent.ACTION_REBOOT;
//重啟設(shè)備時(shí)的廣播
Intent.ACTION_SCREEN_OFF;
//屏幕被關(guān)閉之后的廣播
Intent.ACTION_SCREEN_ON;
//屏幕被打開之后的廣播
Intent.ACTION_SHUTDOWN;
//關(guān)閉系統(tǒng)時(shí)發(fā)出的廣播
Intent.ACTION_TIMEZONE_CHANGED;
//時(shí)區(qū)發(fā)生改變時(shí)發(fā)出的廣播
Intent.ACTION_TIME_CHANGED;
//時(shí)間被設(shè)置時(shí)發(fā)出的廣播
Intent.ACTION_TIME_TICK;
//廣播:當(dāng)前時(shí)間已經(jīng)變化(正常的時(shí)間流逝)譬正。
//當(dāng)前時(shí)間改變宫补,每分鐘都發(fā)送,不能通過組件聲明來(lái)接收曾我,只有通過Context.registerReceiver()方法來(lái)注冊(cè)
Intent.ACTION_UID_REMOVED;
//一個(gè)用戶ID已經(jīng)從系統(tǒng)中移除發(fā)出的廣播
//
Intent.ACTION_UMS_CONNECTED;
//設(shè)備已進(jìn)入U(xiǎn)SB大容量?jī)?chǔ)存狀態(tài)時(shí)發(fā)出的廣播粉怕?
Intent.ACTION_UMS_DISCONNECTED;
//設(shè)備已從USB大容量?jī)?chǔ)存狀態(tài)轉(zhuǎn)為正常狀態(tài)時(shí)發(fā)出的廣播?
Intent.ACTION_USER_PRESENT;
//
Intent.ACTION_WALLPAPER_CHANGED;
//設(shè)備墻紙已改變時(shí)發(fā)出的廣播
詳細(xì)見菜鳥教程http://www.runoob.com/w3cnote/android-tutorial-broadcastreceiver.html