BroadcastReceiver总结

BroadcastReceiver

为了方便进行系统级别通知,android引入了一套广播消息机制,也就是BroadcastReceiver,在Android开发中BroadcastReceiver的使用是非常广泛的它也是Android的四大组件之一,翻译成汉语就是:”广播接收者”。 Android 系统里定义了各种各样的广播,如电池的使用状态,电话的接收和短信的接收,开机启动都会产生一个广播。当然用户也可以自定义自己的广播,它的使用也非常灵活,于是想就BroadcastReceiver做一番总结,来重新了解下android中广播机制。

广播使用

  • 系统广播
    1 . 定义好广播接收器继承BroadcastReceiver,注册广播动作。
    2 . 系统在发出Action时,在广播接收器onReceive接收广播的Action并作出处理。不同的广播对于一组不同的Action。
    3 . 在manifest.xml文件中注册广播,或者通过registerReceiver方法注册广播,当然unregisterReceiver可以注销广播。

  • 自定义广播
    1 . 通过 sendBroadCast 或 sendOrderedBroadcast发送广播
    2 . 接收广播和上面的一样。

广播类型

android广播机制之所以非常灵活,是因为Android应用程序可以对自己感兴趣的广播进行注册,从而只关注自己感兴趣的内容。而在这些内容中可以是系统广播,也可以是其他应用发的广播信息。根据发送者类型可以将广播机制分为:
1.普通广播(Normal Broadcasts)和有序广播(Ordered Broadcasts)。
2. 系统广播和自定义广播。

首先来了解标准广播和有序广播

  • 标准广播

标准广播是一种完全异步的广播机制,发送者在发完广播之后,所有注册了广播的应用都可以同时接受到发送的信息,但是这也就意味着广播无法被拦截。应用发广播时,sendBroadCast表示普通广播,置于用法后面在讲。一般标准的广播流程如下

这里写图片描述

有序广播 是一种同步广播,广播发出之后,前面接收到的广播可以拦截后面的广播,也就是优先级高的广播可以拦截优先级低的广播。当一个广播的逻辑执行完成之后,后面的逻辑才能继续执行,但是要注意的是,不能在onReceiver方法中,执行耗时操作,否则会出现ANR异常,一般在该onReceiver方法中,执行页面的跳转逻辑,创建状态栏通知,或者开启服务等功能。一般有序广播的流程如下:

这里写图片描述

系统广播

Android系统内置了很多广播,方便用户注册实时接收系统信息状态变化。下面是一些见到的系统广播(来自http://blog.csdn.net/tmj2014/article/details/7815455):

Intent.ACTION_AIRPLANE_MODE_CHANGED;
//关闭或打开飞行模式时的广播

Intent.ACTION_BATTERY_CHANGED;
//充电状态,或者电池的电量发生变化
//电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册

Intent.ACTION_BATTERY_LOW;
//表示电池电量低

Intent.ACTION_BATTERY_OKAY;
//表示电池电量充足,即从电池电量低变化到饱满时会发出广播

Intent.ACTION_BOOT_COMPLETED;
//在系统启动完成后,这个动作被广播一次(只有一次)。

Intent.ACTION_CAMERA_BUTTON;
//按下照相时的拍照按键(硬件按键)时发出的广播

Intent.ACTION_CLOSE_SYSTEM_DIALOGS;
//当屏幕超时进行锁屏时,当用户按下电源按钮,长按或短按(不管有没跳出话框),进行锁屏时,android系统都会广播此Action消息

Intent.ACTION_CONFIGURATION_CHANGED;
//设备当前设置被改变时发出的广播(包括的改变:界面语言,设备方向,等,请参考Configuration.java)

Intent.ACTION_DATE_CHANGED;
//设备日期发生改变时会发出此广播

Intent.ACTION_DEVICE_STORAGE_LOW;
//设备内存不足时发出的广播,此广播只能由系统使用,其它APP不可用?

Intent.ACTION_DEVICE_STORAGE_OK;
//设备内存从不足到充足时发出的广播,此广播只能由系统使用,其它APP不可用?

Intent.ACTION_DOCK_EVENT;
//
//发出此广播的地方frameworks\base\services\java\com\android\server\DockObserver.java

Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;
////移动APP完成之后,发出的广播(移动是指:APP2SD)

Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;
//正在移动APP时,发出的广播(移动是指:APP2SD)

Intent.ACTION_GTALK_SERVICE_CONNECTED;
//Gtalk已建立连接时发出的广播

Intent.ACTION_GTALK_SERVICE_DISCONNECTED;
//Gtalk已断开连接时发出的广播

Intent.ACTION_HEADSET_PLUG;
//在耳机口上插入耳机时发出的广播

Intent.ACTION_INPUT_METHOD_CHANGED;
//改变输入法时发出的广播

Intent.ACTION_LOCALE_CHANGED;
//设备当前区域设置已更改时发出的广播

Intent.ACTION_MANAGE_PACKAGE_STORAGE;
//

Intent.ACTION_MEDIA_BAD_REMOVAL;
//未正确移除SD卡(正确移除SD卡的方法:设置--SD卡和设备内存--卸载SD卡),但已把SD卡取出来时发出的广播
//广播:扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)

Intent.ACTION_MEDIA_BUTTON;
//按下"Media Button" 按键时发出的广播,假如有"Media Button" 按键的话(硬件按键)

Intent.ACTION_MEDIA_CHECKING;
//插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
Intent.ACTION_MEDIA_EJECT;
//已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播?
//广播:用户想要移除扩展介质(拔掉扩展卡)。
Intent.ACTION_MEDIA_MOUNTED;
//插入SD卡并且已正确安装(识别)时发出的广播
//广播:扩展介质被插入,而且已经被挂载。
Intent.ACTION_MEDIA_NOFS;
//
Intent.ACTION_MEDIA_REMOVED;
//外部储存设备已被移除,不管有没正确卸载,都会发出此广播?
// 广播:扩展介质被移除。
Intent.ACTION_MEDIA_SCANNER_FINISHED;
//广播:已经扫描完介质的一个目录
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
//
Intent.ACTION_MEDIA_SCANNER_STARTED;
//广播:开始扫描介质的一个目录

Intent.ACTION_MEDIA_SHARED;
// 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
 Intent.ACTION_MEDIA_UNMOUNTABLE;
//
Intent.ACTION_MEDIA_UNMOUNTED
// 广播:扩展介质存在,但是还没有被挂载 (mount)。
Intent.ACTION_NEW_OUTGOING_CALL;

Intent.ACTION_PACKAGE_ADDED;
//成功的安装APK之后
//广播:设备上新安装了一个应用程序包。
//一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
 Intent.ACTION_PACKAGE_CHANGED;
//一个已存在的应用程序包已经改变,包括包名
Intent.ACTION_PACKAGE_DATA_CLEARED;
//清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)
//用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_INSTALL;
//触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
//
Intent.ACTION_PACKAGE_REMOVED;
//成功的删除某个APK之后发出的广播
//一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_REPLACED;
//替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
Intent.ACTION_PACKAGE_RESTARTED;
//用户重新开始一个包,包的所有进程将被杀死,所有与其联系的运行时间状态应该被移除,包括包名(重新开始包程序不能接收到这个广播)
Intent.ACTION_POWER_CONNECTED;
//插上外部电源时发出的广播
Intent.ACTION_POWER_DISCONNECTED;
//已断开外部电源连接时发出的广播
Intent.ACTION_PROVIDER_CHANGED;
//

Intent.ACTION_REBOOT;
//重启设备时的广播

Intent.ACTION_SCREEN_OFF;
//屏幕被关闭之后的广播

Intent.ACTION_SCREEN_ON;
//屏幕被打开之后的广播

Intent.ACTION_SHUTDOWN;
//关闭系统时发出的广播

Intent.ACTION_TIMEZONE_CHANGED;
//时区发生改变时发出的广播

Intent.ACTION_TIME_CHANGED;
//时间被设置时发出的广播

Intent.ACTION_TIME_TICK;
//广播:当前时间已经变化(正常的时间流逝)。
//当前时间改变,每分钟都发送,不能通过组件声明来接收,只有通过Context.registerReceiver()方法来注册

Intent.ACTION_UID_REMOVED;
//一个用户ID已经从系统中移除发出的广播
//

Intent.ACTION_UMS_CONNECTED;
//设备已进入USB大容量储存状态时发出的广播?

Intent.ACTION_UMS_DISCONNECTED;
//设备已从USB大容量储存状态转为正常状态时发出的广播?

Intent.ACTION_USER_PRESENT;
//

Intent.ACTION_WALLPAPER_CHANGED;
//设备墙纸已改变时发出的广播

接下来看下系统广播具体用法

1.动态注册

public class MainActivity extends AppCompatActivity {

    private MyReceiver myReceiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        IntentFilter intentFilter = new IntentFilter("android.net.conn.CONNECTIVITY_CHANGE");
        myReceiver = new MyReceiver();
        registerReceiver(myReceiver, intentFilter);

    }

    public class MyReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent) {
            ConnectivityManager manager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
            NetworkInfo activeNetworkInfo = manager.getActiveNetworkInfo();
            if(activeNetworkInfo != null && activeNetworkInfo.isAvailable()){
                Toast.makeText(MainActivity.this,  "有网", Toast.LENGTH_SHORT).show();
            }else {
                Toast.makeText(MainActivity.this,  "无网", Toast.LENGTH_SHORT).show();
            }
        }
    }

    /**
     * Activity销毁时要注销广播,否则可能内存溢出
     */
    @Override
    protected void onDestroy() {
        super.onDestroy();
        unregisterReceiver(myReceiver);
    }
}

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

打开Menu->System setting -> Data usage切换Mobile Data来启动和禁止手机网络(不是无线网,测试时一直开着无线网,导致一直没反应,此时无线网要关闭)

效果如图:

这里写图片描述

这里写图片描述

2 . 静态注册

广播接收器
public class BootReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context ,"开机完成", Toast.LENGTH_LONG).show();
    }
}

清单文件中注册:

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

    <receiver android:name=".BootReceiver">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
        </intent-filter>
    </receiver>

结果:

这里写图片描述

到此系统广播用法差不多就是这样,静态注册与动态注册也已经基本了解。我们一般应用的动态注册可以自由控制注册和注销,但是当应用完全退出时,应用就接受不到广播了,要想接收广播必需打开应用,这也是它的主要缺点。但静态注册即使应用没有开启,也能接收广播,因为广播的注册在清单文件中,android系统只要解析了清单文件,即使不打开应用,也可以接受广播。

自定义广播

有时候我们,并不需要接受系统广播,而是根据需求实现自定义广播,这样可以实现进程通信(IPC)和线程通信。举例: 当你在后台播放一首音乐时,而前台要显示进度条进度,此时就需要实现服务和Activity之间通信,当然通信有很多种方式,但是我们可以向前台定时发广播,从而实时更新进度条信息,这样很方便实现解耦和。

  • 普通广播

举例看下用法:

我只举例动态注册,静态和系统广播一样就不多说了

//注册:

//action可以自定义
IntentFilter intentFilter = new    
IntentFilter("com.chen.android.MyReceiver");
myReceiver = new MyReceiver();
registerReceiver(myReceiver, intentFilter);

//发广播:
sendBroadcast(new Intent("com.chen.android.MyReceiver"));

//接收
public void onReceive(Context context, Intent intent) {

Toast.makeText(MainActivity.this,  "接收到了自定义广播", Toast.LENGTH_SHORT).show();
}

结果如图:
这里写图片描述

同时你也可以在Intent中携带一些参数信息,比如歌曲进度信息,信息类别和Activity中参数传递一样,就不多说了。

Intent intent = new Intent("com.chen.android.MyReceiver");
intent.putExtra("str", "我在发广播");
sendBroadcast(intent);

这里写图片描述

  • 有序广播
//第一个接收器打印
Toast.makeText(context,  "first receiver", Toast.LENGTH_SHORT).show();

//第二个接收器打印
Toast.makeText(context,  "sencond receiver", Toast.LENGTH_SHORT).show();

//清单文件注册
<receiver android:name=".MyReceiver">
    <intent-filter android:priority="2">
        <action android:name="com.chen.android.MyReceiver" />
    </intent-filter>
</receiver>

<receiver android:name=".MyReceiver2">
    <intent-filter android:priority="1">
        <action android:name="com.chen.android.MyReceiver" />
    </intent-filter>
</receiver>

//发送有序广播
Intent intent = new Intent("com.chen.android.MyReceiver");
sendOrderedBroadcast(intent, null);

其中 priority代表广播优先级,越高越先被接收,并且拥有拦截低优先级的权限
可以看见运行结果:先打印了“first receiver”, 再打印了“sencond receiver”toast信息,当我在第一个接收器中调用abortBroadcast(),运行结果可知只打印了first receiver的Toast信息,证明了前面那张有序广播的图解。

  • 无序广播

就不多做演示了,挺简单的。。。

本地广播

发生和接收的广播全部都是属于系统全局广播,且发出的广播可以被其他任何应用程序接收到,并且可以接收到其他任何应用程序的广播,这样容易引发应用程序的安全性,比如通过广播向应用程序发送垃圾信息,以及发送的一些携带关键性数据的广播有可能被其他应用拦截。这些都是不允许的,在android中引入了一套广播机制,使用这个广播接收器只能接收来自本应用的广播,其他的接收不到,这样就不容易引发系统安全问题,本地广播使用了一个LocalBroadcastmanager对广播进行管理,并提供对广播的注销和注册,看下它的api。

这里写图片描述

1 .其中getInstance获取LocalBroadcastmanager
2 .registerReceiver注册广播
3 .unregisterReceiver注销广播
4 .sendBroadcast 发送广播

这是常用的api,可以知道 它和动态注册广播的使用方式一样,就不多说了。

最后

本篇总结借鉴了郭大侠的《第一行代码中》的广播章节代码,具体可以查看书籍。

1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 、4下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。、可私 6信博主看论文后选择购买源代码。 1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 、4下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。、可 6私信博主看论文后选择购买源代码。 1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 、4下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。、可私 6信博主看论文后选择购买源代码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值