Android笔记--BroadcastReceiver
BroadcastReceiver(一)--用法总结#
一. BroadcastReceiver简介
BroadcastReceiver是Android四大组件之一,他的作用是监听并且接受系统或者其他App发出的广播,并对感兴趣的广播做处理。
二. 广播的应用场景
Android系统中各个组件之间的通信。
同一应用或者不同应用都可以做到。最开始的组件化解耦常用的办法就是利用广播。但是后续出来了更优秀的解耦框架比如ARouter等
对Android系统发出的特定广播进行监听。并做出相应的处理。
例如,看视频的时候,如果意外wifi断掉,该为了4G网络。那这种情况采用监听网络状态的广播就很实用。利用网络状态的变化对视频进行暂停,给用户一个有好的提示。
多线程通信
三. 内部流程
实现设计三个方面的内容:
- 广播接收方
- 广播发送方
- AMS
三者中,广播接收方和广播发送方不直接交互。而是通过AMS利用Binder通信完成。接收方和发送方不会感知对方的存在。
要想完成一个完整的广播发送和接受的流程要按一下的过程执行:
- 广播接收方(BroadcastReceiver)注册到AMS中(Binder通信)
- 广播发送方(sendBroadcast)发送广播到AMS(Binder通信)
- AMS会在自己所在的system_server进程中维护一个广播接收器的注册列表,当广播发送方将消息发送过来以后,AMS会根据广播发送方的IntentFilter和permission等信息在自己维护的注册表中匹配合适的广播接受器。一旦匹配成功,AMS就会利用Binder把广播消息发送到对应的广播接收器的消息循环队列。
- 广播接收器会在自己的消息循环对列中拿广播消息,并回调onReceive方法。
四. BroadcastReceiver实例构建
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//重写onReceive方法。当接收到广播以后会回调这个方法
}
}
其中需要注意onReceive方法不能执行耗时操作。
广播接收器注册
广播接收器注册有两种方式:
- 静态注册
- 动态注册
静态注册方式
和activity在manifest.xml的存在形式类似:
<receiver
android:enabled=["true" | "false"]
/*
指定是否启用了接收机,也就是说,是否可以被系统实例化
*/
android:exported=["true" | "false"]
/*
*exproted表示这个组件对于其他应用的可见性。可选值是true false
true | 能被外部应用调用,但是可以通过permission选项做权限过滤
fasle | 正好相反
默认值:
如果没有intent filter标签:
那么activity, receiver, service: 默认 false
如果有intent filter标签:
那么activity, receiver, service: 默认 true
不管有没有intent filter标签:
contentProvider总是true
*
/
android:icon="drawable resource"
android:label="string resource"
android:name=".MyBroadcastReceiver"//receiver的名称
android:permission="string"//这个属性用于权限验证,限定只有启动本组件的Intent具有相同的permission值传过来,才能成功启动本组建。适用于四大组件
android:process="string"
//BroadcastReceiver运行所处的进程
//默认为app的进程,可以指定独立的进程
//注:Android四大基本组件都可以通过此属性指定自己的独立进程
>
//用于指定此广播接收器将接收的广播类型
//本示例中给出的是用于接收网络状态改变时发出的广播
<intent-filter>
<action android:name="这里可以自己编写,也可以写系统发的广播名称" />
</intent-filter>
</receiver>
动态注册方式
在代码中调用context.registerReceiver()方法即可
// 选择在Activity生命周期方法中的onResume()中注册
@Override
protected void onResume(){
super.onResume();
// 1. 实例化BroadcastReceiver子类 & IntentFilter
MyBroadcastReceiver mBroadcastReceiver = new MyBroadcastReceiver();
IntentFilter intentFilter = new IntentFilter();
// 2. 设置接收广播的类型为自定义的广播
intentFilter.addAction("com.demo.testaction");
// 3. 动态注册:调用Context的registerReceiver()方法
registerReceiver(mBroadcastReceiver, intentFilter);
}
// 注册广播后,要在相应位置记得销毁广播
// 当此Activity实例化时,会动态将该广播注册到系统中
// 当此Activity销毁时,动态注册的广播将不再接收到相应的广播。
@Override
protected void onPause() {
super.onPause();
//销毁广播
unregisterReceiver(mBroadcastReceiver);
}
注意:动态广播要在onResume中注册,在onPause中注销
如果不注销会导致内存泄露,重复注销和重复注册也是不允许的。在onResume()注册、onPause()注销是因为onPause()在App死亡前一定会被执行,从而保证广播在App死亡前一定会被注销,从而防止内存泄露。假设我们将广播的注销放在onStop(),onDestory()方法里的话,有可能在Activity被销毁后还未执行onStop(),onDestory()方法,即广播仍还未注销,从而导致内存泄露。但是,onPause()一定会被执行,从而保证了广播在App死亡前一定会被注销,从而防止内存泄露
两种注册方式的区别:
静态注册:伴随App启动而注册到AMS,在App存活状态下静态注册广播常驻内存中,不受其他组件的影响,一旦有感兴趣的广播到来就可以接受。
动态注册:在代码中调用context.registerReceiver()方法才注册进AMS,而且会跟随context的周期而存活。context结束即广播接收器结束。不能做到整个App存活状态接受广播。
两者的实质区别就是注册的方式,和存活的时间。
发送广播
发送刚播使用context.sendBroadcast方法发送,其中重要参数是Intent,而Intent是广播消息的载体
广播类型
现在上面介绍了广播接收器分类和写法,广播如何发送。接下来介绍广播实体有哪些类型
- 普通广播
- 有序广播
- 系统广播
- App内部广播
- 粘性广播
普通广播
普通广播是最常用的一种:
Intent intent = new Intent();
//这里写要发送的广播的动作
intent.setAction(BROADCAST_SAMPLE);
sendBroadcast(intent)
对应的广播接收器如果要接受这个广播,就需要在action中指明
<receiver android:name=".MyBroadcastReceiver">
<intent-filter>
<action android:name="BROADCAST_SAMPLE"/>
</intent-filter>
</receiver>
有序广播
这种有序广播发出去以后,对于一些列接收者有先后循序要求。
规则是这样的:
- 广播接收者们按照priority属性指定的大小排序。值大的先接收到广播。
- priority相同,那动态注册的先接受广播
其中:
- 广播接收者们按顺序依次收到广播。
- 先收到的接收器可以拦截消费掉/修改广播内容,消费掉以后后续接收器不会再收到广播。修改之后后续接收器收到的是修改过的广播。
使用方法:
sendOrderedBroadcast
系统广播
系统广播的发送者是系统本身。比如:
屏幕被关闭 Intent.ACTION_SCREEN_OFF
屏幕被打开 Intent.ACTION_SCREEN_ON
等等,都是系统发出的广播。作为系统广播,开发者只能按需实现接收器
App内部广播
App内部广播可以防止第三方得知intent-filter标签内容以后骚扰性地发广播给我们的应用。
App内部广播可理解为一种局部广播,广播的发送者和接收者都同属于一个App。
相比于全局广播(普通广播),App应用内广播优势体现在:安全,效率高
使用方法1:
- 注册广播标签中exported属性指定为false,这样就不会把这个组件暴露给外部应用
- 发送接受广播都设置权限
- 发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中
利用intent.setPackage(packageName)指定报名
使用方法2:
使用封装好的LocalBroadcastManager类
使用方式上与全局广播几乎相同,只是注册/取消注册广播接收器和发送广播时将参数的context变成了LocalBroadcastManager的单一实例
注:对于LocalBroadcastManager方式发送的应用内广播,只能通过LocalBroadcastManager动态注册,不能静态注册
粘性广播(5.0以后已失效)
粘性广播会滞留,不仅保证在广播发送之前注册了的接收器可以收到广播,也保证后续注册了该粘性广播的接收器也可以接收到这个广播。这种广播已经在5.0失效
这里需要注意的是permission的使用:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.demo.viewdemo">
<permission android:name="com..viewdemo.receivermusthave"/><!-- 这里生命的是发送者必须携带的权限 -->
<permission android:name="com.demo.viewdemo.sendermusthave"/><!-- 这里声明的是接收者必须携带的权限 -->
<uses-permission android:name="com.demo.viewdemo.receivermusthave"/><!-- uses-permission 是代表本应用拥有的权限,而发送者要求的权限,本App必须具备才可以 -->
<uses-permission android:name="com.demo.viewdemo.sendermusthave"/><!-- 发送广播的应用必须具备这个权限,我们自己的应用才会去接收 -->
<application>
<receiver android:name=".MyBroadcastReceiver"
android:permission="com.demo.viewdemo.sendermusthave">
<!-- permission属性是要求发送者必须具备的权限 -->
<intent-filter>
<action android:name="con.demo.TT"/>
</intent-filter>
</receiver>
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>
还需要注意的是在同一个应用里,
这种要求发送方必须具备的权限,即使省略,依旧可以收到这个广播。
但是最好都声明出来。避免不必要的麻烦
Android笔记--BroadcastReceiver的更多相关文章
- Android 学习笔记 BroadcastReceiver广播...
PS:不断提升自己,是件好事... 学习内容: 1.BroadcastReceiver的使用.. 2.通过BroadcastReceiver去启动Service... 1.BroadcastRecei ...
- Android笔记三十三.BroadcastReceiver使用
广播是一种广泛运用在应用程序之间传输信息的机制,而BroadcastReceiver是对发送出来的广播进行过滤接收并响应的一类组件. BroadcastReceiver本质上是一种全局监听器. ...
- Android笔记(六十)Android总结:四大组件——BroadcastReceiver篇
什么是BroadcastReceiver BroadcastReceiver是Android体系的四大组件之一,本质上是一种全局的监听器,用于监听系统全局的广播消息,正式因为其本质为全局监听,因此可以 ...
- Android笔记(二十六) Android中的广播——BroadcastReceiver
为了方便进行系统级别的消息通知,Android有一套类似广播的消息机制,每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收自己所关心的广播内容,这些广播可能是来自于系统,也可能是来自于 ...
- Android学习笔记BroadcastReceiver(广播接收者)
Android发送广播的过程 代码实现 MainActivity.java import androidx.appcompat.app.AppCompatActivity; import androi ...
- Android中BroadcastReceiver的两种注册方式(静态和动态)详解
今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式. BroadcastReceiver也就是"广播接收者"的意思,顾名思义,它就是用来 ...
- Android 广播 BroadcastReceiver
Android 系统里定义了各种各样的广播,如电池的使用状态,电话的接收和短信的接收,开机启动都会产生一个广播.当然用户也可以自定义自己的广播. 既然说到广播,那么必定有一个广播发送者,以及广播接收器 ...
- Android中BroadcastReceiver广播
BroadCastReceiver 简介 广播接收者( BroadcastReceiver )用于接收广播 Intent ,广播 Intent 的发送是通过调用 Context.sendBroadca ...
- java攻城狮之路(Android篇)--BroadcastReceiver&Service
四大组件:activity 显示. contentProvider 对外暴露自己的数据给其他的应用程序.BroadcastReceiver 广播接收者,必须指定要接收的广播类型.必须明确的指定acti ...
随机推荐
- bzoj4763
$分块$ $一个很有趣的技巧$ $在树上选sqrt(n)个关键点,每两个关键点之间的距离<=sqrt(n),每个关键点属于一条链$ $预处理出每两个关键点的bitset$ $每次询问就暴力向上爬 ...
- 错误: 实例 "ahwater-linux-core" 执行所请求操作失败,实例处于错误状态。: 请稍后再试 [错误: Exceeded maximum number of retries. Exceeded max scheduling attempts 3 for instance 7c1609c9-9d0f-4836-85b3-cefd45f942a7. Last exception: [u
错误: 实例 "ahwater-linux-core" 执行所请求操作失败,实例处于错误状态.: 请稍后再试 [错误: Exceeded maximum number of ret ...
- android apk 防止反编译技术第二篇-运行时修改字节码
上一篇我们讲了apk防止反编译技术中的加壳技术,如果有不明白的可以查看我的上一篇博客http://my.oschina.net/u/2323218/blog/393372.接下来我们将介绍另一种防止a ...
- html中连续点击某个标签会出现蓝色的解决方法
给标签加上下面的属性就可以了,也可以把这些属性建立一个class名,谁需要的时候加上也ok -moz-user-select: none; /*mozilar*/ -webkit-user-selec ...
- [xdoj1227]Godv的数列(crt+lucas)
解题关键:1001=7*11*13,模数非常小,直接暴力lucas.递归次数几乎为很小的常数.最后用中国剩余定理组合一下即可. 模数很小时,一定记住lucas定理的作用 http://acm.xidi ...
- RHEL6安装JDK7
一.安装准备 1.操作系统:redhat-server-6.1-x86_64 下载地址: http://www.verycd.com/files/d39b97540497d24175340915244 ...
- Unobtrusive Javascript有三层含义
一是在HTML代码中不会随意的插入Javsscript代码,只在标签中加一些额外的属性值,然后被引用的脚本文件识别和处理: 二是通过脚本文件所增加的功能是一种渐进式的增强,当客户端不支持或禁用了Jav ...
- Automake使用(初级)
工程地址 https://github.com/silvermagic/automakeDev.git 最初工程目录结构 $ vim main.cpp $ vim src/main.cpp ls -l ...
- 51nod1153(dfs/单调队列)
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1153 题意:中文题诶- 思路:一个比较简单的方法是dfs隐式 ...
- solidity 学习笔记(5)接口
接口:不用实现方法,仅仅定义方法. pragma solidity ^; contract cat{ //cat实际上实现了接口animalEat,因为他们有相同的方法. string name; f ...