廣播機制簡介
Android
中的廣播機制更加靈活,因為Android
中的每個應用程序都可以對自己感興趣的廣播進行注冊,這樣該程序就只會接收到自己所關心的廣播內容,這些廣播可能是來自系統的,也可能是來自于其他應用程序的。Android
提供了一套完整的API,允許應用程序自由地發送或者接受廣播。發送廣播的方法其實之前有稍微提到過有一下,就是借助第二章的Intent
。接受廣播需要用到一個新的工具,廣播接收器。
Android
中的廣播可以分為兩類,標準廣播和有序廣播。
標準廣播
標準廣播是一種完全異步執行的廣播。在廣播發出之后,所有的廣播接收器幾乎都會在同一時刻接收到這條廣播信息 ,因此它們之間沒有任何先后順序可言。這種廣播效率較高,但是同時也意味著它是無法被階段的。
有序廣播
有序廣播是一種同步執行的廣播。在廣播發出之后,同一時刻只有一個廣播接收器能夠收到這條廣播消息。當這個廣播接收器中的邏輯執行完畢之后廣播才會繼續傳遞。優先級高的廣播接收器就可以先收到廣播消息,并且前面的廣播接收器還可以截斷正在傳遞的廣播。
接收系統廣播
Android
內置了很多系統級別的廣播,我們可以在應用程序張通過監聽這些廣播來得到各種系統的狀態信息。比如手機開機、電池電量發生變化、時間或者時區發生改變等等。如果想要接收到這些廣播就需要使用廣播接收器。
動態注冊監聽網絡變化
廣播接收器可以自由地對自己感興趣的廣播進行豬兒,這樣當有相應的廣播發出時,廣播接收器就能夠收到該廣播,并且在內部處理相應的邏輯。
注冊廣播的方式一般也有兩種,在代碼中注冊或者在AndroidManifest.xml
中注冊,其中前者也被稱為動態注冊,后者就是靜態注冊。
如何創建呢?其實只需要新建一個類,讓它繼承自BroadcastReceiver
,并且重寫父類的onReceive
方法就行了。這樣當有廣播到來時,onReceive
方法就會得到執行,具體的邏輯就可以在這個方法中處理。
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private NetworkChangeReceiver networkChangeReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intentFilter = new IntentFilter();
networkChangeReceiver = new NetworkChangeReceiver();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
registerReceiver(networkChangeReceiver, intentFilter);
}
@Override
protected void onDestroy(){
super.onDestroy();
unregisterReceiver(networkChangeReceiver);
}
class NetworkChangeReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context, "network changes", Toast.LENGTH_SHORT).show();
}
}
}
可以看到,我們在MainActivity
中定義了一個內部類NetworkChangeRectiver
,這個類繼承自BroadcastReceiver
,重寫了父類的onReceive
方法,這樣每當網絡狀態發生變化,onReceive
方法就會得到執行。
觀察一下onCreate
方法,我們創建了一個IntentFilter
實例,并給他添加了一個值為android.net.conn.CONNECTIVITY_CHANGE
的action
。當系統網絡狀態發生變化是,系統發出的正是一條值為android.net.conn.CONNECTIVITY_CHANGE
的廣播。我們的廣播接收器想要監聽什么廣播,就在這里添加相應的action
即可。接下來創建了一個NetworkChangeReceiver
的實例,然后調用registerReceiver
方法進行注冊,將NetworkChangeReceiver
的實例和IntentFilter
的實例都傳了進去,這樣NetworkChangeRecevier
就會受到所有值為android.net.conn.CONNECTIVITY_CHANGE
的廣播。
最后記得,動態注冊的廣播接收器一定都要取消注冊才行,這里我們是在onDestroy
方法中通過調用unregisterReceiver
來實現的。
細化調整
只是提醒網絡發生了變化還是不夠人性化,最好能準確地告訴用戶當前有網絡還是沒有。因此需要進一步的優化。
public void onReceive(Context context, Intent intent){
ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
if (networkInfo != null && networkInfo.isAvailable())
Toast.makeText(context, "network is available", Toast.LENGTH_SHORT).show();
else
Toast.makeText(context, "network is unavailable", Toast.LENGTH_SHORT).show();
}
這里直接監控系統網絡,需要在AndroidManifest.xml
里面注冊一下權限。
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
訪問 開發者手冊 可以查看Android
系統所有可以聲明的權限。
靜態注冊實現開機啟動
動態注冊的廣播接收器可以自由地控制祖冊預祝校,在靈活性方面有很大的優勢,但是也存在著一個缺點,就是必須要在程序啟動之后才能接收到廣播,因為注冊的邏輯是寫在onCreate
方法中的。如果要讓程序在未啟動的情況就能接收到廣播,就需要使用靜態注冊的方式了。
這里讓程序及收一條開機廣播,當收到這條廣播,就可以在onReceive
方法里執行相應的邏輯,從而實現開機啟動的功能。
創建一個BootCompleteReceiver
類。
public class BootCompleteReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();
}
}
這里不再使用內部類的方式來定義廣播接收器。我們再將這個接收器注冊到AndroidManifest.xml
中,將這個廣播接收器的類名注冊進去。
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.oujitsune.broadcasttest">
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<application
...
<receiver android:name=".BootCompleteReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
</application>
</manifest>
在<application>
標簽內出現了一個新的標簽<receiver>
,所有靜態注冊的廣播接收器都是在這里注冊的。用法和<activity>
標簽非常相似,首先通過android:name
來指定具體注冊哪一個廣播接收器,然后再<intent-filter>
標簽里加入想要接受的廣播就行了。
另外,監聽系統開機廣播也是需要聲明權限的。
發送自定義廣播
接下來我們來看看怎么在應用程序中發送自定義的廣播。
發送標準廣播
在發送廣播之前,我們還是需要先定義一個廣播接收器來接受廣播才行。
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_LONG).show();
}
}
注冊一下。
我們一會兒要發的就是com.example.broadcasttest.MY_BROADCAST
。
<receiver android:name=".MyBroadcastReceiver">
<intent-filter>
<action android:name="com.example.broadcasttest.MY_BROADCAST"/>
</intent-filter>
</receiver>
發送有序廣播
廣播是一種跨進程的通信方式,這一點從前面接收系統廣播的時候就可以看出來了。因此我們在應用程序內發出的廣播,其他的應用程序也是可以收到的。
想要發送有序廣播,只需要修改一行代碼。
將onClick
方法中改為sendOrderedBroadcast(intent, null);
。
這個方法接受兩個參數,第一個仍然是Intent
,第二個是與權限相關的字符串。
如何設定廣播接收器的先后順序呢?自然是在注冊的時候設定了。
修改AndroidManifest.xml
中的代碼:
<intent-filter android:priority="100">
,即在intent-filter
標簽下修改優先級為100。
然后在接收器重寫的onReceive
方法中調用abortBroadcast
方法,這個方法截斷接收到的廣播。
使用本地廣播
前面我們發送和接受的廣播都是系統的全局廣播,發出的廣播可以被其他任何應用程序接收到。這樣容易引起安全問題,比如我們發送一些攜帶關鍵性數據的廣播,可能被其他應用程序接貨,或者其他的程序不停地想廣播接收器里發送各種垃圾廣播。
為了解決廣播的安全性問題,Android
引入了一套本地廣播的機制,使用這個機制發出的廣播只能在應用程序內部進行床底,并且廣播接收器也只能接受來自本應用程序發出的廣播,這樣就不會有安全性問題了。
本地廣播并不復雜,主要就是使用了一個LocalBroadcastManager
來對廣播進行管理,并且提供了發送廣播和注冊廣播接收器的方法。下面我們就通過具體的實例來嘗試一下。
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private LocalReceiver localReceiver;
private LocalBroadcastManager localBroadcastManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
...
localBroadcastManager = localBroadcastManager.getInstance(this);
Button button_broadcast = (Button) findViewById(R.id.button_broadcast);
button_broadcast.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.example.broadcasttest.LOCAL_BROADCAST");
localBroadcastManager.sendBroadcast(intent);
}
});
intentFilter = new IntentFilter();
intentFilter.addAction("com.example.broadcasttest.LOCAL_BROADCAST");
localReceiver = new LocalReceiver();
localBroadcastManager.registerReceiver(localReceiver, intentFilter);
}
@Override
protected void onDestroy(){
super.onDestroy();
localBroadcastManager.unregisterReceiver(localReceiver);
}
class LocalReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context, "received local broadcast", Toast.LENGTH_LONG).show();
}
}
}
上面的代碼基本和原先用的動態注冊接收器和發送廣播是一樣的,只不過現在首先是通過LocalBroadcastManager
的getInstance
方法得到了它的一個實例,然后再注冊廣播接收器的時候調用的是LocalBroadcastManager
的registerReceiver
方法,在發送廣播的時候調用的是LocalBroadcastManager
的sendBroadcast
方法。
另外注明:本地廣播的無法通過靜態注冊的方法來接收的。
最后再來盤點一下本地廣播的優勢吧:
- 可以明確地指定正在發送的廣播不會離開我們的程序,因此不用擔心數據泄露。
- 其他程序無法將廣播發送到我們程序內部,因此不需要擔心會有安全漏洞的隱患。
- 發送本地廣播相比發送全局廣播更加高效。
實踐——實現強制下線功能
強制下線功能只需要彈出一個對話框,讓用戶只能點擊確定按鈕,回到登錄界面。為了避免需要在每一個活動中添加一個對話框,用廣播實現是一個好辦法。
關閉所有的活動只需要用AcitivityCollector
類來管理所有的活動,然后用BaseActivity
類作為所有活動的父類。
首先我們創建一個LoginActivity
作為登錄界面,用表格布局來實現一下布局就可以了。
<?xml version="1.0" encoding="utf-8"?>
<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:stretchColumns="1">
<TableRow>
<TextView
android:layout_height="wrap_content"
android:text="Account: "/>
<EditText
android:id="@+id/account"
android:layout_height="wrap_content"
android:hint="Input you account"/>
</TableRow>
<TableRow>
<TextView
android:layout_height="wrap_content"
android:text="Password: " />
<EditText
android:id="@+id/password"
android:layout_height="wrap_content"
android:inputType="textPassword"/>
</TableRow>
<TableRow>
<Button
android:id="@+id/login"
android:layout_height="wrap_content"
android:layout_span="2"
android:text="Login"/>
</TableRow>
</TableLayout>
public class LoginActivity extends BaseActivity {
private EditText accountEdit;
private EditText passwordEdit;
private Button login;
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.login);
accountEdit = (EditText) findViewById(R.id.account);
passwordEdit = (EditText) findViewById(R.id.password);
login = (Button) findViewById(R.id.login);
login.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String account = accountEdit.getText().toString();
String password = passwordEdit.getText().toString();
if (account.equals("254886715") && password.equals("Zh980728xl")){
Intent intent = new Intent(LoginActivity.this, FirstActivity.class);
startActivity(intent);
finish();
} else
Toast.makeText(LoginActivity.this, "account or password is invalid", Toast.LENGTH_LONG).show();
}
});
}
}
在LoginActivity
里判斷一下賬號密碼,如果正確就進入FirstActivity
。
后面的活動我們只要實現一下強制下線就可以了。
添加一個強制下線按鈕,發送一條強制下線廣播,然后重寫一個廣播接收器就可以實現了。
//廣播接收器
public class GetOutReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context, Intent intent){
AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
dialogBuilder.setTitle("Warning");
dialogBuilder.setMessage("You are forced to be offline. Please try to login again.");
dialogBuilder.setCancelable(false);
dialogBuilder.setPositiveButton("OK", 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();
}
}
這個廣播接收器里顯然不只是之前僅僅一個Toast
那么簡單,還加入了AlertDialog.Builder
來構建一個對話框,注意一定要調用setCancelable
方法將對話框設置為不可取消,然后使用setPositiveButton
方法來給對話框注冊確定程序,當用戶點擊了確定按鈕,就調用ActivityCollect
的finishAll
方法來銷毀所有活動,因此一定要給Intent
加入FLAG_ACTIVITY_NEW_TASK
這個標志,最后還需要把對話框里的類型設置為TYPE_SYSTEM_ALERT
,不然會無法在廣播接收器里彈出。
這樣,主要邏輯都完成了。還需要注冊一下。
這里有幾點需要注意,首先由于我們在ForceOfflineReceiver
里彈出了一個系統級別的對話框,因此必須要進行聲明android.permission.SYSTEM_ALERT_WINDOW
權限,然后對LoginActivity
進行注冊,并且把它設置為主活動。