一、广播是什么
在Android中,它是一种广泛运用在应用程序之间传输信息的机制,Android中我们发送广播内容是一个Intent,这个Intent中可以携带我们要发送的数据。
二、广播的使用场景
- 同一app之间不同进程之间通信
- 不同app之间组件间的通信
三、广播的种类
- 标准广播:一种完全异步执行的广播,广播发出之后所有广播接收器几乎会在同一时刻收到该广播,他们之间没有任何顺序。
context.sendBroadcast(Intent)
方法发送的广播,不可被拦截 - 有序广播:一种同步执行的广播,在广播发出后同一时刻只会有一个广播接收器收到该广播,当这个广播接收器执行完相应逻辑后,广播才会继续传递,有先后顺序,优先级高的广播就可以先接收到广播消息并且可以截断这个广播。
context.sendOrderBroadcast(Intent)
方法发送的广播,可被拦截 - 本地广播:
localBroadcastManager.sendBroadcast(Intent)
只在app内传播
四、广播接收器
广播接收器是专门用来接收广播信息的,分为静态注册和动态注册两种:
- 静态注册:注册完成后就一直在运行
Enabled属性:是否启用这个广播接收器
Exported属性:是否允许该广播接收器接收本程序以外的广播
public class BootCompleteReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"广播接收器",Toast.LENGTH_SHORT).show();
}
}
静态注册的广播接收器一定要在AndroidManifest.xml文件中注册后才可以使用,注册代码如下所示:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.strive.broadcast">
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
</application>
</manifest>
- 动态注册广播:跟随Activity的生命周期
优点:动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大优势
缺点:必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在onCreate()方法中
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private NetWorkChangeRecriver netWorkChangeRecriver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intentFilter = new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
netWorkChangeRecriver = new NetWorkChangeRecriver();
//注册广播接收器
registerReceiver(netWorkChangeRecriver,intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
//取消注册广播接收器
unregisterReceiver(netWorkChangeRecriver);
}
class NetWorkChangeRecriver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
//获取系统服务类,专门用于管理网络连接
ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
if(!("".equals(networkInfo))&&networkInfo.isAvailable()){
Toast.makeText(context, "网络连接正常", Toast.LENGTH_SHORT).show();
}else{
Toast.makeText(context, "网络连接失败", Toast.LENGTH_SHORT).show();
}
}
}
}
五、广播内部实现机制
- 自定义广播接收者BroadcastReceiver,并且重写onReceiver()方法。
- 通过Binder机制向AMS(Activity Manager Service)进行注册。
- 广播发送者通过Binder机制向AMS发送广播。
- .AMS查找符合条件(IntentFilter/Permission等)的BroadcastReceiver,将广播发送到相应的BroadcastReceiver(一般情况下是Activity)的消息队列中。
- 消息循环执行拿到此广播,回调BroadcastReceiver中的onReceiver()方法。
六、本地广播
为了解决广播安全性的问题,Android引入了一套本地广播机制,使用这个机制发出的广播只能够在应用程序内部进行传递,并且广播接收器也只能接受应用程序内部的广播。
本地广播的发送和注册广播接收器都需要使用LocalBroadcastManager来对广播进行管理,并提供发送广播和注册广播接收器的方法。
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private LocalBroadcastManager localBroadcastManager;
private LocalReceiver localReceier;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
localBroadcastManager = LocalBroadcastManager.getInstance(this);
intentFilter = new IntentFilter();
intentFilter.addAction("com.example.broadcasttest.LOCAL_BROADCAST");
localReceier = new LocalReceiver();
localBroadcastManager.registerReceiver(localReceier,intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
localBroadcastManager.unregisterReceiver(localReceier);
}
class LocalReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "localbroadcast", Toast.LENGTH_SHORT).show();
}
}
}
本地广播的特点
- 使用它发送的广播将只在自身app内传播,因此你不必担心泄漏隐私的数据。
- 其他app无法对你的app发送该广播,因此你的app根本不可能收到非自身app发送的该广播,因此你不必担心有安全漏洞可以利用。
- 发送本地广播比发送系统广播更加的高效。
本地广播的内部实现机制
- LocalBroadcast高效的原因:因为它内部是通过Handler实现的,它的sendBroadcast()方法含义并非和系统的sendBroadcast()一样,它的sendBroadcast()方法其实就是通过Handler发送了一个Message而已。
- LocalBroadcast安全的原因:既然它是通过Handler实现广播发送的,那么相比系统广播通过Binder机制实现那肯定更加高效,同时使用Handler来实现,别的app无法向我们应用发送该广播,而我们app内部发送的广播也不会离开我们的app。
- LocalBroadcast内部协作主要是靠两个Map集合:mReceivers和mActions,当然还有一个List集合mPendingBroadcasts,这个主要存储待接收的广播对象。