0.0
BroadcastReceiver是android中的四大组件之一。意思是广播接收者,运用的场合比较多。Android中的广播分为两种:标准广播和有序广播。
标准广播(Normal Broadacsts):也可以理解为无序广播。标准广播的发送是异步的,所有广播接收者都会在同一时间接收到广播信息。对于广播接收者广播接收者来说,既不能截断广播,也不能修改广播。
有序广播(Ordered Broadcasts):相对于标准广播而言。有序广播的发送是同步的,广播接收者会按照优先级的高低而先后接收到广播,也就是说,优先级高的广播接收者会先接收到广播,优先级低的广播接收者后接收到广播。而且重要的一点是,优先级高的广播接收者在收到广播后,具有截断和修改广播的权限。
0.1 广播接收者的注册方式
BroadcastReceiver有两种注册方式:动态注册与静态注册。
动态注册: 也就是在代码中进行注册。实际注册可参看下面代码示例(这里以监听网络变化为例):
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
NetworkChangeReceiver networkChangeReceiver = new NetworkChangeReceiver();
registerReceiver(networkChangeReceiver,intentFilter);//动态注册广播接收者
unregisterReceiver(networkChangeReceiver);//取消注册
另外,还有一点必须说明,Android系统为了保证应用程序的安全性,规定在访问某些重要信息时,必须在配置文件中声明权限,否则会导致程序崩溃。在此的监听网络变化也需要声明权限。所以在清单文件AndroidManifest.xml中加入这样一条权限:
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
静态注册: 在AndroidManifest.xml文件中注册。在此以监听开机完成的广播为例(新建一个BootCompleteReceiver类继承BroadcastReceiver),展示下广播接收者的静态注册。
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<application
…
<receiver android:name=".BootCompleteReceiver">
<intent-filter>
<aciton android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
</application>
请注意,以上静态注册的第一行,同样声明了一个接收开机完成广播的权限。
广播接收者的动态注册与静态注册两者相比较,动态注册有其优点,即可以自由地控制注册与注销,比较灵活。但也有缺点,只有在打开程序运行的时候才能接收广播,原因是注册接收广播的逻辑写在onCreate()方法中。反之,一旦退出程序就不能再接收广播了。而静态注册的广播接收者则不存在这个问题。
0.2 自定义广播的发送与接收
0.2.1 标准广播的发送接收
写一个例子来说明。新建项目Broadcast01。为了能够接收到广播,先定义一个广播接收者MyBroadcastReceiver继承自BroadcastReceiver:
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "received broadcast in MyBroadcastReceiver", 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.beijing.broadcast01">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
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=".MyBroadcastReceiver">
<intent-filter>
<action android:name="com.beijing.broadcast01.MY_BROADCAST" />
</intent-filter>
</receiver>**
</application>
</manifest>
上面让MyBroadcastReceiver接受一条值为com.beijing.broadcast01.MY_BROADCAST的广播,所以在发送广播的时候,就需要发出同样的广播。修改activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.beijing.broadcast01.MainActivity">
<Button
android:id="@+id/button"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Send Broadcast" />
</RelativeLayout>
在activity_main.xml中加入了一个Button,点击它来发送一条广播。再来完成MainActivity里的逻辑。:
public class MainActivity extends AppCompatActivity {
@Override
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() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.beijing.broadcast01.MY_BROADCAST");
sendBroadcast(intent);
}
});
}
}
为button设置点击事件,在onClick()方法中借助intent发送一条值为com.beijing.broadcast01.MY_BROADCAST的广播。这样只要是监听这个广播的广播接收者就都能收到广播。运行程序,点击button按钮,效果图如下:
0.2.2 有序广播的发送接收
广播是一种跨进程的通信方式,一个应用程序发出的广播不仅自己可以收到,其他应用程序也可以接收。下面用示例来说明。新建第二个项目Broadcast02,在这个项目中定义一个广播接收者SecondBroadcastReceiver继承自BroadcastReceiver,同样在onReceive()方法中弹出一个toast:
public class SecondBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "received broadcast in SecondBroadcastReceiver", 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.beijing.broadcast02">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
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=".SecondBroadcastReceiver">
<intent-filter>
<action android:name="com.beijing.broadcast01.MY_BROADCAST" />
</intent-filter>
</receiver>
</application>
</manifest>
注意上面注册时,让SecondBroadcastReceiver也监听与MyBroadcastReceiver同样的广播com.beijing.broadcast01.MY_BROADCAST。现在将项目Broadcast02安装到模拟器上,然后再回到Broadcast01界面,点击Send Broadcast按钮看看效果:
先后弹出了两条提示信息,充分说明了一个应用程序发出的广播也可以被其他应用程序接收。以上都是标准广播,下面演示一下有序广播。关闭Broadcast02项目,修改Broadcast01中MainActivity的一行代码sendBroadcast(intent);为下面的代码:
sendOrderedBroadcast(intent,null);
这里的发送有序广播有两个参数,第一个是intent,第二个与权限有关,传入null即可。不过现在运行的话,看不出有什么明显的效果。还需修改一下两个广播接收者的权限,这里将MyBroadcastReceiver的priority的值设为100:
<receiver android:name=".MyBroadcastReceiver">
<intent-filter android:priority="100">
<action android:name="com.beijing.broadcast01.MY_BROADCAST" />
</intent-filter>
</receiver>
这样的话,MyBroadcastReceiver就比SecondBroadcastReceiver的优先级高,会先于后者收到广播。另外,有序广播中,优先级高的接收者可以修改或截断广播。打开MyBroadcastReceiver,在onReceive()方法中添加一行代码:
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "received broadcast in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();
abortBroadcast();
}
}
以上onReceive()方法中调用了abortBroadcast()来阻断广播的向下传递,现在运行Broadcast01并点击按钮,就只会弹出一条提示信息:
0.3 本地广播的运用
前面使用的都是系统全局广播,也就是说,自己发出的广播可以被其他任何应用程序接收,同时自己也能收到其他应用程序的各种广播。这样容易产生安全问题,尤其是机密性的数据与信息。再者,常常收到其他应用程序的垃圾广播,也是个麻烦。因此,Android引入了一套本地广播机制。在这种机制下,广播在不同程序之间是无法互通的, 只能在同一个应用程序中传递。本地广播机制的使用,保证了安全性问题。
本地广播其实主要就是靠LocalBroadcastManager来管理广播,并提供了发送广播和注册广播接收者的方法。下面在Broadcast01的基础上用例子来说明。
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private LocalBroadcastManager localBroadcastManager;
private LocalReceiver localReceiver;
@Override
protected 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() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.beijing.broadcast01.LOCAL_BROADCAST");
localBroadcastManager.sendBroadcast(intent);
}
});
intentFilter = new IntentFilter();
intentFilter.addAction("com.beijing.broadcast01.LOCAL_BROADCAST");
localReceiver = new LocalReceiver();
localBroadcastManager.registerReceiver(localReceiver, intentFilter);//注册本地广播监听器
}
@Override
protected void onDestroy() {
super.onDestroy();
localBroadcastManager.unregisterReceiver(localReceiver);
}
private class LocalReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "received local broadcast", Toast.LENGTH_SHORT).show();
}
}
}
如前面所说,本地广播只是使用了LocalBroadcastManager来管理广播。上面的代码中,在发送广播和注册广播监听器的时候,前面都加上了LocalBroadcastManager。运行并点击按钮效果如下:
值得注意的是,本地广播是不能用静态注册的。原因是静态注册即使应用程序不启动也能接收广播,而本地广播在发送广播时就已经启动,所以根本无需静态注册。