BroadcastReceive基础知识总结

BroadcastReceive基础知识总结

1、BroadcastReceive简介

  • BroadcastReceive也就是“广播接收者”的意思,顾名思义,就是用来接收来自系统和应用中的广播
  • 在Android系统中,广播体现在方方面面,例如当开机完成后系统会产生一条广播,接收到这条广播就能实现开机启动服务的功能,当网络状态改变时,系统会产生一条广播,接收到这条广播,就能及时的做出提示和保存数据等操作,当电池的电量改变的时候,系统会产生一条广播,接收到这条广播就能在电量低的时候告知用户,及时保存进度。等等
  • Android中的广播机制设计的非常出色,很多事情原本需要开发者亲自操作的额,现在只需要广播告知自己就可以了,大大的减少了开发的工作量和开发周期,而作为应用开发者,就需要熟练掌握Android系统提供的一个开发利器,那就是BroadcastReceive。

2、广播接收器的类型

  • (1)Normal broadcasts:默认广播

    • 发送一条默认的广播 Context.sendBroadcast()方法,普通广播对于多个接收者来说是异步的,通常每个接受者都无需等待即可以接收到广播,接受者相互之间不会有影响,对于这种广播,接受者无法终止广播,即无法阻止其他接受者的接收动作
    • MainActivity.java
        sendNormal = (Button) findViewById(R.id.sendNormal);
        sendNormal.setOnClickListener(new OnClickListener());

    /**
     * 自定义的内部click
     */
    class OnClickListener implements View.OnClickListener {

        @Override
        public void onClick(View v) {
            switch (v.getId()) {
                case R.id.sendNormal:
                    Intent intent = new Intent("com.wuyinlei.action.BROADCAST");
                    intent.putExtra("info","我们很有缘,谢谢你收到我");
                    MainActivity.this.sendBroadcast(intent);
                    break;
            }
        }
    }

MyReceiver.java


/**
 * 自定义的广播接收器
 */
public class MyReceiver extends BroadcastReceiver {
    public MyReceiver() {
    }

    /**
     * 接受的方法
     * @param context
     * @param intent
     */
    @Override
    public void onReceive(Context context, Intent intent) {
        String info = intent.getStringExtra("info");
        Toast.makeText(context, info, Toast.LENGTH_SHORT).show();
    }
}

mainfest.xml


        " data-snippet-id="ext.c74b9bdb5284d8870d9b36ec3110c416" data-snippet-saved="false" data-codota-status="done"> <receiver
            android:name=".MyReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.wuyinlei.action.BROADCAST"></action>
            </intent-filter>
        </receiver>
  • (2)Ordered broadcasts:有序广播

    • 发送一个有序广播使用Context.sendOrderedBroadcast()方法,有序广播比较特殊,它每次只发死你个到优先级较高的接受者那里,然后由优先级高的接受者在传播到优先级别低的接受者那里,优先级高的接受者又能力终止这个广播
  • (3)Sticky Broadcast:粘性广播
    • 当处理完之后的Intent,依然存在,知道你把它去掉

3、广播接收器的创建步骤

  • 1、构建Intent,使用sendBroadcast方法发送广播
  • 2、定义一个广播接收器,该广播接收器集成BroadcastReceiver,并且覆盖onReceive()方法来响应事件
  • 3、注册广播接收器,我们可以在代码中注册,也可以在AndroidManifest.xml配置文件中注册

4、注册广播接收器的两种方式

  • 静态注册

    • 静态注册是在AndroidManifest.xml文件中配置
    
        " data-snippet-id="ext.bfe5c8cd2eaef0a38610405c077a48e0" data-snippet-saved="false" data-codota-status="done">             <receiver
            android:name=".MyReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.yinlei.action.MY_BROADCAST"></action>
            </intent-filter>
        </receiver>
            这个时候,我们需要在Activity或者Service中这样
    
             Intent intent = new Intent("com.yinlei.action.MY_BROADCAST");
                            intent.putExtra("info","我是自定义广播");
                            sendBroadcast(intent);
  • 动态注册

    • 动态注册需要在代码中动态的指定广播地址并注册,通常我们是在Activity或者Service注册一个广播
                MyReceiver receiver = new MyReceiver();
                IntentFilter filter = new IntentFilter();
                filter.addAction("android.intent.action.MY_BROADCAST");
                registerReceiver(receiver,filter);

                解除注册
                unregisterReceiver(receiver);
  • 注意:

    • 在这个方法中发来的广播中,代码注册方式中,接收到的广播的先后和注明优先级最高的他们的先后是随机的,如果没有优先级,代码注册收到为最先

5、有序广播

  • 发送广播
sendOrderedBroadcast();
中使用android:priority属性,这个属性的范围在-1000---1000。
* 数值越大,优先权越高,在广播接收者中使用setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接受者哪里,这样优先级低的接受者可以用getResultExtras获取到最新的经过处理的信息集合
* 使用sendOrderedBroadcast方法有序广播的时候,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null,则表示接收者要接收此广播,需声明指定权限,这样做是从安全角度考虑的,例如系统的短信就是有序的广播的形式,一个应用可能是具有拦截垃圾短信的功能,当短信到来的时候它可以先接收到短信,必要时候终止广播的传递,这样的软件就必须声明接收短信的权限。
" data-snippet-id="ext.bebc66240b73b4bfe80d76873b8e4df3" data-snippet-saved="false" data-codota-status="done">* 在注册广播中的<intent-filter>中使用android:priority属性,这个属性的范围在-1000---1000。
* 数值越大,优先权越高,在广播接收者中使用setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接受者哪里,这样优先级低的接受者可以用getResultExtras获取到最新的经过处理的信息集合
* 使用sendOrderedBroadcast方法有序广播的时候,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null,则表示接收者要接收此广播,需声明指定权限,这样做是从安全角度考虑的,例如系统的短信就是有序的广播的形式,一个应用可能是具有拦截垃圾短信的功能,当短信到来的时候它可以先接收到短信,必要时候终止广播的传递,这样的软件就必须声明接收短信的权限。
  • 终止广播传递
abortBroadcast()
 * 同级别接收是先后是随机的,在到级别低的收到广播,如果先接收到的把广播拦截了,同级别以外的接收者是无法收到该广播的
 * 在这个方法发来的广播(代码注册方式中),收到广播先后次序为:注明优先级的、代码优先级的、没有优先级的,如果都没有优先级,代码注册接收到优先
  • 创建的两个有序广播
case R.id.sendOrder:
                    Intent intentOrder = new Intent("com.wuyinlei.action.MY_BROADCAST");

                    //第一个参数是intent,第二个是接收的权限
                    MainActivity.this.sendOrderedBroadcast(intentOrder,null);
                    break;

mainfest.xml配置


        " data-snippet-id="ext.4c5096240514e62de5b15f1ecb77672a" data-snippet-saved="false" data-codota-status="done"> <receiver
            android:name=".ThreeReceiver"
            android:enabled="true"
            android:exported="true"
            >
            <intent-filter android:priority="100">
                <action android:name="com.wuyinlei.action.MY_BROADCAST"></action>
            </intent-filter>
        </receiver>
        <receiver
            android:name=".FourReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="200">
                <action android:name="com.wuyinlei.action.MY_BROADCAST"></action>
            </intent-filter>
        </receiver>

有序广播的gif图:

有序广播传值gif图:

6、粘性广播

  • 发送粘性广播使用:sendStickBroadcast(intent)
  • 发送这个广播需要权限
" data-snippet-id="ext.2190c8bd341940dfb81d567957c97d62" data-snippet-saved="false" data-codota-status="done"> <uses-permission android:name="android.permission.BROADCAST_STICKY"/>
  • 去掉这个removeStickyBroadcast(intent)
  • sendStickyOrderedBroadcast():这个方法具有有序广播的特性也有粘性广播的特性;
    • 在这个方法发来的广播中,代码注册方式中,收到广播先后次序为:注明优先级的、代码注册的、没有优先级的;如果都没有优先级,代码注册的收到最先。

      MainActivity.java

                case R.id.sendSticky:
                    Intent intentSticky = new Intent("com.wuyinlei.action.MY_BROADCAST_STICKY");
                    MainActivity.this.sendStickyBroadcast(intentSticky);
                    break;

                case R.id.startSticky:
                    startActivity(new Intent(MainActivity.this, ReceiveActivity.class));
                    break;

ReceiveActivity.java

 @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_receive);
        mFiveReceiver = new FiveReceiver();
    }

    @Override
    protected void onResume() {
        super.onResume();
        IntentFilter filter = new IntentFilter();
        filter.addAction("com.wuyinlei.action.MY_BROADCAST_STICKY");
        registerReceiver(mFiveReceiver,filter);
    }

    @Override
    protected void onPause() {
        super.onPause();
        unregisterReceiver(mFiveReceiver);
    }

粘性广播GIF图:

###7、接收系统广播

* 1、开机启动广播

* 我们经常会有这样的应用场合,比如消息推送服务,需要实现开机启动的功能,要实现这个功能,我们就可以订阅系统“启动完成”这条广播,接收到这条广播,我们就可以启动自己的服务了


                " data-snippet-id="ext.f2827e64caf189531d0ae95f7274e3fb" data-snippet-saved="false" data-codota-status="done">              <intent-filter>
                    <!--注册开机广播地址-->
                    <action android:name="android.intent.action.BOOT_COMPLETED"/>
                    <category android:name="android.intent.category.DEFAULT"/>
                </intent-filter>

这个开机广播地址,从安全角度考虑,系统要求必须声明接收开机启动广播的权限。

" data-snippet-id="ext.fb919e36d72a156d09fa219c179e9a06" data-snippet-saved="false" data-codota-status="done">            <user-permission android:name="android.peimission.RECEIVE_BOOT_COMPLETE"/>
* 2、网络状态变化
   * 比如用户游览游览器信息的时候,网络突然断开,我们要及时的提醒用户网络已经断开,要实现这个功能,我们可以接受网络状态改变这样的一条广播,当由连接状态变为断开状态时候,系统就会收到一条广播,我们接收到之后,在通过网络的状态做出相应的操作

" data-snippet-id="ext.7470d917ff305b674e7b2fa2de46d31b" data-snippet-saved="false" data-codota-status="done">         <intent-filter>
                <action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
                <category android:name="android.intent.category.DEFAULT"/>
            </intent-filter>
访问的网络状态权限:

 ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
    NetworkInfo info = cm.getActiveNetworkInfo();
    String typeName = info.getTypeName();//连接类型
    info.getState();//连接状态
" data-snippet-id="ext.4ea6103e2191c0014a18553676cbdc01" data-snippet-saved="false" data-codota-status="done"> <user-permission android:name="android.peimission.ACCESS_NETWORK_STATE"/>

 ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
    NetworkInfo info = cm.getActiveNetworkInfo();
    String typeName = info.getTypeName();//连接类型
    info.getState();//连接状态
* 3、电量变化
   * 如果我们在使用阅读软件,可能全屏阅读,这个时候用户就看不到剩余的电量了,我们就可以为他们提供电量的信息,要想做到这一点,我们需要接受一条电量变化的广播,然后获取百分比信息
   * int currLevel = intent.getIntExtra(BatteryManager.EXTRA_LEVEL,0)  //当前电量
   * int total = intent.getIntExtra(BatteryManager.EXTRA_SCALE,1)//总电量
   * int percent = currLevel*100/total;

" data-snippet-id="ext.e8ec24372a4676644baec1d3dba5af7b" data-snippet-saved="false" data-codota-status="done">              <intent-filter>
            <action android:name="android.intent.action.BATTERY_CHANGED"/>
            <category android:name="android.intent.category.DEFAULT"/>
                            </intent-filter>
   立即获取电量,而不是等电量变化的广播,可以使用:
       Intent batteryIntent = getApplicationContext().registerReceiver(null,new IntentFilter(Intent.ACTION_BATTERY_CHANGED))

“`

这里附上github地址:https://github.com/wuyinlei/Broadcast

时间: 03-31

BroadcastReceive基础知识总结的相关文章

MySQL数据库基础知识

day02 MySQL数据库基础知识 一.基础知识概述: 基础决定你这门课程的学习成败!只有学习好这些基础知识以后,你才能真正的运用自如.才能够对数据库有更深入的了解,道路才会越走越远. 二.基础知识: 1.数据库(database):数据库就好比是一个物理的文档柜,一个容器,把我们整理好的数据表等等归纳起来. 创建数据库命令:        create database 数据库名; 2.查看数据库         show databases; 3.打开指定的数据库         use 

linux入门基础知识及简单命令介绍

linux入门基础知识介绍 1.计算机硬件组成介绍 计算机主要由cpu(运算器.控制器),内存,I/O,外部存储等构成. cpu主要是用来对二进制数据进行运算操作,它从内存中取出数据,然后进行相应的运算操作.不能从硬盘中直接取数据. 内存从外部存储中取出数据供cpu运存.内存的最小单位是字节(byte) 备注:由于32的cpu逻辑寻址能力最大为32内存单元.因此32位cpu可以访问的最大内存空间为:4GB,算法如下: 2^32=2^10*2^10*2^10*2^2 =1024*1024*1024

基础知识--:before伪元素和:after伪元素

http://book.51cto.com/art/201108/285688.htm 3.7  替换指定位置 大家都知道before和after是前.后的意思.但是奇怪的是,CSS中的:before伪元素和:after伪元素是为源文档中不存在的内容设置样式的. 没有内容怎么设置样式呢?别急!它们有一个content属性,一起使用就可以为某个选择器前.后的内容设置样式了. 下面就来了解一下:before伪元素和:after伪元素的用法. 视频教学:光盘/视频/3/3.7  替换指定位置.avi 

20_Shell语言———VIM编辑器基础知识三之窗口属性定制、配置文件及查找替换功能

Vim编辑器可以让用户按照需求来定制一些使用属性. 一.窗口属性定义 1)显示行号 行号不是内容,只是用来帮助用户确认文本所在的行.在vim编辑器中,如果要显示行号,可以在末行模式下输入: set number 如果想关闭,则可以在功能名称前面加上no,即: set nonumber 命令可以被简写,如set number 可以简写为 set nu:set nonumber 可以简写为 set nonu. 注意,上述设定仅对当前vim的进程有效,一旦当前进程关闭,这些设定就会失效,如果要使设定永

web基础知识(一)关于ajax传值最基础东西

HTTP方法之 GET对比POST GET:从指定的资源请求数据, POST:向指定的资源提交要被处理的数据 GET方法: 请注意,查询字符串(名称/值对)是在 GET 请求的 URL 中发送的: /test/demo_form.asp?name1=value1&name2=value2 有关 GET 请求的其他一些注释: GET 请求可被缓存 GET 请求保留在浏览器历史记录中 GET 请求可被收藏为书签 GET 请求不应在处理敏感数据时使用 GET 请求有长度限制 GET 请求只应当用于取回

线程基础知识

什么是线程: 在一个程序里的一个执行路线就叫做线程(thread).更准确的定义是:线程是"一个进程内部的控制序列" 一切进程至少都有一个执行线程 进程与线程 进程是资源竞争的基本单位 线程是程序执行的最小单位 线程共享进程数据,但也拥有自己的一部分数据 线程ID 一组寄存器 栈 errno 信号状态 优先级 fork和创建新线程的区别 当一个进程执行一个fork调用的时候,会创建出进程的一个新拷贝,新进程将拥有它自己的变量和它自己的PID.这个新进程的运行时间是独立的,它在执行时几乎

Keepalived基础知识

大纲: 一.什么是Keepalived? 二.VRRP协议简介. 三.Keepalived原理. 四.Keepalived配置文件详解. 五.Keepalived配置示例. 一.什么是Keepalived? 什么是Keepalived呢,keepalived观其名可知,保持存活,在网络里面就是保持在线了,也就是所谓的高可用或热备,用来防止单点故障(单点故障是指一旦某一点出现故障就会导致整个系统架构的不可用)的发生,那说到keepalived时不得不说的一个协议就是VRRP协议,可以说这个协议就是

【Python数据挖掘课程】六.Numpy、Pandas和Matplotlib包基础知识

前面几篇文章采用的案例的方法进行介绍的,这篇文章主要介绍Python常用的扩展包,同时结合数据挖掘相关知识介绍该包具体的用法,主要介绍Numpy.Pandas和Matplotlib三个包.目录:        一.Python常用扩展包        二.Numpy科学计算包        三.Pandas数据分析包        四.Matplotlib绘图包 前文推荐:       [Python数据挖掘课程]一.安装Python及爬虫入门介绍       [Python数据挖掘课程]二.K

【深度分解】听趣拍云产品经理剖析视频基础知识(2)

"随着技术的不断进步,视频技术的制作加工门槛逐渐降低,信息资源的不断增长,同时由于视频信息内容更加丰富完整的先天优势,在近年来已经逐渐成为主流.在基础知识(1)里面已经讲了模拟时代和数字化时代的视频技术.接下来将对视频编码与压缩.画面压缩.运动压缩.互联网视频应用的到来做一个详细的介绍." 视频编码与压缩 视频编码与压缩,是数字化视频非常重要的技术,以至于它直接影响到视频在各个领域的应用.如果没有视频编码技术的不断提高,我们今天也不可能在方方面面享受到视频的便利性. 首先,视频编码是一