BroadcastReceiver粗解

来源:互联网 发布:算法导论读书报告 编辑:程序博客网 时间:2024/06/04 22:21

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。运行并点击按钮效果如下:

这里写图片描述

值得注意的是,本地广播是不能用静态注册的。原因是静态注册即使应用程序不启动也能接收广播,而本地广播在发送广播时就已经启动,所以根本无需静态注册。

原创粉丝点击