Android BroadcastReceiver 广播深入研究

Android深入探究笔记之二十 — 广播接收者,BroadcastReceiver

1. 简单概述

       广播被分为两种不同的类型:“普通广播(Normal broadcasts)、有序广播(Ordered broadcasts)、粘性消息.

       普通广播是完全异步的,可以在同一时刻(逻辑上)被所有接收者接收到,消息传递的效率比较高。

       但缺点是:接收者不能将处理结果传递给下一个接收者,并且无法终止广播Intent的传播。

       有序广播是按照接收者声明的优先级别,被接收者依次接收广播。广播传输过程可以被终止,使用setResultExtras(Bundle),接受用Bundle bundle = getResultExtras(true))获取上一个数据。

        缺点是:容易被截取,面试纪要。

       粘性消息在发送后就一直存在于系统的消息容器里面,等待对应的处理器去处理,如果暂时没有处理器处理这个消息则一直在消息容器里面处于等待状态。       

         以下是调用方法:

      Context.sendBroadcast()  --------普通消息发送方式

      Context.sendOrderedBroadcast()--------有序广播发送方式   

       sendStickyBroadcast()--------有序广播发送方式

2.生命周期详解:

   在Android中,每次广播消息到来时都会创建BroadcastReceiver实例并执行onReceive() 方法,

       onReceive() 方法执行完后,BroadcastReceiver 的实例就会被销毁。

       当onReceive() 方法在10秒内没有执行完毕,Android会认为该程序无响应。

       所以在BroadcastReceiver里不能做一些比较耗时的操作,否侧会弹出ANR(Application No Response)的对话框。

       如果需要完成一项比较耗时的工作,应该通过发送Intent给Service,可以启动独立服务,或者启动activity来做长时间操作。

       这里不能使用子线程来解决,因为 BroadcastReceiver 的生命周期很短,子线程可能还没有结束BroadcastReceiver就先结束了。

       BroadcastReceiver一旦结束,此时BroadcastReceiver所在的进程很容易在系统需要内存时被优先杀死,

       因为它属于空进程(没有任何活动组件的进程)。

       如果它的宿主进程被杀死,那么正在工作的子线程也会被杀死。所以采用子线程来解决是不可靠的。

3.注册两种方式:

1:  静态注册是指将注册文件放入到配置文件中进行注册

        实现过程:

        ①:继承BroadcastReceiver 并复习void onReceive(Context context, Intent intent)

        ②:将广播注册到manifest.xml中          

[java] 
view plain
copy
print
?

  1.  <receiver android:name=“BroastRev”>  // BroadcastReceiver  子类的名字(处理者)  
  2.             <intent-filter>  
  3.              <action android:name=“com.pdw.activity.ACTION”></action> //对应的ACTION  
  4.             </intent-filter>  
  5. </receiver>  

        ③:发送广播

[java] 
view plain
copy
print
?

  1. <pre class=“java” name=“code”>intent.setAction(ACTION);  //<span style=”color:#ff0000;”>对应配置文件中的ACTION = “com.pdw.activity.ACTION “</span>  
  2. dBroadcast(intent);</pre><br>  

       机制分析:

         当触发了sendBroadcast方法后系统会到manifest.xml中寻找与ACTION相对应的action ,找到后会去实例化里面对应的recevier也就是这里的BroastRev,然后执行里面的void onReceive(Context context, Intent intent)方法

2:动态注册是指在程序中注册广播

      实现过程:

       ①:实例化一个IntentFilter 并注册如下

[java] 
view plain
copy
print
?

  1. receiver = new BroastRev();  //BroadcastReceiver 子类的名字(处理者)  
  2. filter = new IntentFilter(); //实例化IntentFilter   
  3. filter.addAction(ACTION);  
  4. MainActivity.this.registerReceiver(receiver, filter); //注册  

     ②:广播

[java] 
view plain
copy
print
?

  1. intent.setAction(ACTION);  //<span style=”color:#ff0000;”>对应IntentFilter 中的ACTION = “com.pdw.activity.ACTION “</span>  
  2. sendBroadcast(intent);  

     ③:解除注册

[java] 
view plain
copy
print
?

  1. MainActivity.this.unregisterReceiver(receiver);  //调用unregisterReceiver解除注册 这里的receiver 与注册的时候的是同一个对象  

      机制分析:动态注册会再程序上保持一个ACTION然后当发出广播的时候会在程序上搜索是否存在相对应的ACTION 如果存在则进入到广播子类中去!

3:优缺点分析

    1、 在代码中用函数代码动态的方式注册。动态注册的处理器必须用代码动态的销毁,每次用来处理消息的就一个实例对象

    2、 在配置文件里面静态注册,静态注册有个特点,那就是一旦注册就会一直存在于系统里面,无论应用是否关闭或开关机。(简直就是一个流氓软件病毒啊~)。静态注册每次有处理消息就由系统new一个处理器处理,并销毁

4:注意事项

    动态绑定只是实现了绑定,所以还是要进行发送了广播才能进行接收。也就是说BroastRev 里面的onReceive方法才会接收到!

    原文作者:尐噺
    原文地址: https://blog.csdn.net/Apple_hsp/article/details/39079217
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞