Android——广播
Android中每个程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自于系统的,也可能是来自于其他应用程序的。Android提供了一套完整的API,允许这些应用自由的发送和接收广播。发送广播借助Intent,接收广播借助广播接收器。
广播类型:主要有两种,标准广播和无序广播。
- 标准广播:完全异步执行的广播,在广播发出后所有广播接收器会几乎在同一时刻接收到这条广播消息。效率高,无法被截断。
- 有序广播:同步执行的广播,再广播发出后,同一时刻只会有一个广播接收器能够接收到这条广播消息。当这个广播接收器中的逻辑执行完毕后,广播才会继续传递,此时广播接收器是有先后顺序的,优先级高的广播接收器就接收到广播消息,并且前面的广播接收器还可以截断正在传递的广播。
接收系统广播
Android内置了许多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。想要接收这些广播就要使用广播接收器。
广播接收器可以对自己感兴趣的广播进行注册,当有相应的广播发出后,广播接收器就可以接收到该条广播,并在内部处理相应的逻辑。注册广播的方式一般分为两种,静态注册和动态注册。
-
动态注册监听网络变化
public class MainActivity extends Activity { private IntentFilter intentFilter; private NetworkChangeReceiver networkChangeReceiver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); intentFilter = new IntentFilter(); intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE"); networkChangeReceiver = new NetworkChangeReceiver(); registerReceiver(networkChangeReceiver,intentFilter) } @Override protected void onDestroy() { super.onDestroy(); unregisterReceiver(networkChangeReceiver); } class NetworkChangeReceiver extends BroadcaBstReceiver { @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context, "network changes", Toast.LENGTH_SHORT).show(); } } }
我们定义了一个内部类NetworkChangeReceiver,这个类继承自BroadcaBstReceiver的,只重写了onReceive() 方法,每当网络状态发生改变时,onReceiver就会得到执行。
在onCreate()方法中,创建了一个IntentFilter的实例,并为这个实例添加了一个值为android.net.conn.CONNECTIVITY_CHANGE 的 action ,这样当网络状态发生变化时,系统发出的正是一个值为android.net.conn.CONNECTIVITY_CHANGE 的广播。也就是我们的广播接收器要接受什么样的广播就在里添加相应的antion就可以了。然后创建一个NetworkChangeReceiver实例,调用registerReceiver()方法进行注册,动态注册的广播接收器一定要取消注册才行,在onDestroy()方法中通过unregisterReceiver()放法进行取消注册。
另外,Android为了保证应用程序的安全性要求如果程序要是访问一些系统的关键信息,必须要在配置信息中声明权限才可以,否则程序将直接崩溃。查询网络状态就是要声明权限的。
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.myapplication"> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> // 声明权限 <!-- ................. --> </manifest>
-
静态注册·实现开机启动
动态注册的广播接收器可以灵活的控制注册与注销,但是必须要在程序启动后才能接收到广播,因为注册逻辑是写在onCreate()里的,若要实现开机自动启动就要使用静态注册的方式了。
让程序接收一条开机广播,当收到这条广播时就可以在onReceice()方法里了执行相应的逻辑,以实现开机启动的功能。
创建一个类继承自BroadcastReceive;
public class BootCompleteReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show(); } }
这里仅用简单的Toast输出一条信息。
后在清单文件中注册,
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <!-- ·············· --> <receiver android:name=".BootCompleteReceiver" > <intent-filter> <action android:name="android.intent.action.BOOT_COMPLETED" /> </intent-filter> </receiver> </application>
标签<receiver> 所有静态注册的广播接收器都是在这里进行注册的。它的用法其实和<activity>标签非常相似,首先通过 android:name来指定具体注册哪一个广播接收器,然后在<intent-filter>标签里加入想要接收的广播就行了。
监听系统开机广播也是需要声明权限的,可以看到,我们使用<uses-permission>标签又加入了一条 android.permission.RECEIVE_BOOT_COMPLETED 权限。
发送自定义广播
-
发送标准广播
-
定义广播接收器(前面有讲;继承 / 注册)
-
发送标准广播:
public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button button = (Button) findViewById(R.id.button); button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent("com.example.broadcasttest.MY_BROADCAST"); sendBroadcast(intent); } }); } }
发送自定义广播的逻辑。首先构建出了一个 Intent 对象,并把要发送的广播的值传入,然后调用了 Context 的 sendBroadcast()方法将广播发送出去,这样所有监听 com.example.broadcasttest.MY_BROADCAST 这条广播的广播接收器就会收到消息。此时发出去的广播就是一条标准广播。
-
-
发送有序广播
广播是一种可以跨进程的通信方式,因此在我们应用程序内发出的广播,其他的应用程序也是可以收到的。
-
定义广播接收器(前面有讲;继承 / 注册)
-
发送有序广播:与无序广播只需要改变一行代码,即将 sendBroadcast()方法改成sendOrderedBroadcast()方法。
sendOrderedBroadcast()方法接收两个参数,第一个参数仍然是Intent,第二个参数是一个与权限相关的字符串, 这里传入 null 就行了。
-
确定广播的先后顺序:在注册的时候就已经设定了,修改AndroidManifest.xml 的代码。
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <!-- ·············· --> <receiver android:name=".BootCompleteReceiver" > <intent-filter android:priority="100"> <action android:name="android.intent.action.BOOT_COMPLETED" /> </intent-filter> </receiver> </application>
通过 android:priority 属性给广播接收器设置优先级,优先级比较高的广播接收器就可以先收到广播。 如果在 onReceive()方法中调用了 abortBroadcast()方法,就表示将这条广播截断 。
-
使用本地广播
前面讲到的全部都是属于系统的全局广播。即发出的广播可以被任何程序接收到,并且也可以接收其他任何应用程序的广播,这样有时候会有安全问题。
Android引入了本地广播的机制,使用这个机制发出的广播只应用内部进行传递,并且广播接收器也只能接收来自本应用程序发出的广播。
本地广播的用法并不复杂,主要就是使用了一个 LocalBroadcastManager 来对广播进行管理。