【Android】广播机制
一、广播机制简介
android的应用程序可以自由的发送和接收广播,这些广播可以来自于系统和其它应用程序,Android提供了一套完整的API,允许应用自由的发送和接收广播。
广播类型
标准广播:
是一种完全异步执行的广播,所有的广播接收器几乎会在同一时间接收到广播消息,无法被截断。
有序广播:
同步执行的广播,同一时刻只有一个广播接收器能够接收到这条广播消息,广播接收器有先后顺序,优先级高的广播接收器先收到广播消息,前面的广播接收器可以截断正在传递的广播。
二、接收系统广播
2.1动态注册
在代码中对自己感兴趣的广播进行注册,动态注册只有在程序启动后才能接收到广播
以监听网络变化为例:
1.新建一个类,继承BroadcastReceiver并重写父类onReceive方法,当有广播时会执行这个方法,具体的逻辑就在这个方法中处理
2.在onCreate方法中创建IntentFilter实例,并给它添加相应广播的action
3.创建新建类的实例,调用registerReceiver方法进行注册,将新建类实例和IntentFilter实例都传进去
public class MainActivity extends AppCompatActivity {private IntentFilter intentFilter;private NetworkChangeReceiver networkChageReceiver;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);intentFilter=new IntentFilter();intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");networkChageReceiver=new NetworkChangeReceiver();registerReceiver(networkChageReceiver,intentFilter);}protected void onDestory(){super.onDestroy();unregisterReceiver(networkChageReceiver);}class NetworkChangeReceiver extends BroadcastReceiver{public void onReceive(Context context, Intent intent){Toast.makeText(context,"network changes",Toast.LENGTH_SHORT).show();}}
}
动态注册的广播会在onDestroy中调用unregisterReceiver方法取消注册
class NetworkChangeReceiver extends BroadcastReceiver{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,"available",Toast.LENGTH_SHORT).show();}else {Toast.makeText(context,"unavailable",Toast.LENGTH_SHORT).show();}}}
使用getSystemService得到的ConnectivityManager实例是一个系统服务类,专门用于管理网络连接,之后调用getActiveNetworkInfo方法得到NetworkInfo实例,之后调用NetworkInfo的isAvailable可以判断出当前是否有网络 。
2.2静态注册
静态注册可以可以在程序未启动的情况下接收广播
1.创建一个广播接收器,修改onReceive方法。
Exported属性表示是否允许这个广播接收器接收本程序以外的广播
Enabled属性表示是否启用这个广播接收器
public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// TODO: This method is called when the BroadcastReceiver is receiving// an Intent broadcast.Toast.makeText(context,"Boot Complete",Toast.LENGTH_SHORT).show();}
}
静态广播接收器要在AndroiManifest.xml中注册
<manifest xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/><applicationandroid:allowBackup="true"android:dataExtractionRules="@xml/data_extraction_rules"android:fullBackupContent="@xml/backup_rules"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:roundIcon="@mipmap/ic_launcher_round"android:supportsRtl="true"android:theme="@style/Theme.BroadcastTest"tools:targetApi="31"><receiverandroid:name=".BootCompleteReceiver"android:enabled="true"android:exported="true"><intent-filter><action android:name="android.intent.action.BOOT_COMPLETED"/></intent-filter></receiver><activityandroid:name=".MainActivity"android:exported="true"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity></application></manifest>
<receiver>用法和<activity>标签相似,通过android:name来指定具体注册哪一个广播接收器
还要在 <intent-filter>标签里添加相应的action
三、发送自定义广播
3.1发送标准广播
新建广播接收器
public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// TODO: This method is called when the BroadcastReceiver is receiving// an Intent broadcast.Toast.makeText(context,"MyBroadcastReceiver",Toast.LENGTH_SHORT).show();}
}
修改广播接收器注册
<receiverandroid:name=".MyBroadcastReceiver"android:enabled="true"android:exported="true"><intent-filter><action android:name="com.example.broadcasttest.MY_BROADCAST"/></intent-filter></receiver>
修改主活动代码,发送广播
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button button=(Button) findViewById(R.id.button);button.setOnClickListener(new View.OnClickListener(){public void onClick(View v){Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");intent.setPackage(getPackageName());sendBroadcast(intent);}});}
Android 8.0+ 的特殊处理
对于 Android 8.0 (API 26) 及以上版本:
必须为静态注册的接收器指定
package
: intent.setPackage(getPackageName());
3.2发送有序广播
程序发送的广播可以被其他的应用程序接收
发送有序广播需要修改主活动
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button button=(Button) findViewById(R.id.button);button.setOnClickListener(new View.OnClickListener(){public void onClick(View v){Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");intent.setPackage(getPackageName());sendOrderedBroadcast(intent,null);}});}
使用 sendOrderedBroadcast,接收两个参数Intent和与权限有关的字符串
在注册时可以设置广播接收器的先后顺序
<receiverandroid:name=".MyBroadcastReceiver"android:enabled="true"android:exported="true"android:permission="your.custom.permission"><intent-filter android:priority="100"><action android:name="com.example.broadcasttest.MY_BROADCAST"/></intent-filter></receiver>
设置优先级后,优先级高的广播接收器可以选择是否允许广播继续传递
如果在onReceive方法中调用abortBroadcast(),就表示将这条广播截断
3.3本地广播
本地广播机制下发出的广播只能在应用程序的内部进行传播,广播接收器也只能接收来自本应用程序发出的广播。
使用LocalBroadcastManager对广播进行管理,提供了发送广播和注册广播接收器的方法。
public class MainActivity extends AppCompatActivity {private IntentFilter intentFilter;private LocalReceiver receiver;private LocalBroadcastManager localBroadcastManager;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);localBroadcastManager=LocalBroadcastManager.getInstance(this);Button button=(Button) findViewById(R.id.button);button.setOnClickListener(new View.OnClickListener(){public void onClick(View v){Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");localBroadcastManager.sendBroadcast(intent);}});intentFilter=new IntentFilter();intentFilter.addAction("com.example.broadcasttest.MY_BROADCAST");receiver=new LocalReceiver();localBroadcastManager.registerReceiver(receiver,intentFilter);}protected void onDestroy() {super.onDestroy();// 必须取消注册,否则内存泄漏!if (receiver != null) {unregisterReceiver(receiver);}}class LocalReceiver extends BroadcastReceiver{@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context,"local",Toast.LENGTH_SHORT).show();}}
}
本地广播无法通过静态注册的方式接收
本地广播的优势
1.不担心机密数据泄露
2.其他程序无法将广播发送到程序内部,不担心安全漏洞隐患
3.发送本地广播比系统全局广播更高效