深入Android中BroadcastReceiver的两种注册方式(静态和动态)详解

今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式。

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

在我们详细分析创建BroadcastReceiver的两种注册方式前,我们先罗列本次分析的大纲:

(1)对静态和动态两种注册方式进行概念阐述以及演示实现步骤

(2)简述两种BroadcastReceiver的类型(为后续注册方式的对比做准备)

(3)在默认广播类型下设置优先级和无优先级情况下两种注册方式的比较

(4)在有序广播类型下两种注册方式的比较

(5)通过接受打电话的广播,在程序(Activity)运行时和终止运行时,对两种注册方式的比较

(6)总结两种方式的特点

第一步:静态和动态注册方式基本概念以及实现步骤

构建Intent,使用sendBroadcast方法发出广播定义一个广播接收器,该广播接收器继承BroadcastReceiver,并且覆盖onReceive()方法来响应事件注册该广播接收器,我们可以在代码中注册(动态注册),也可以AndroidManifest.xml配置文件中注册(静态注册)。

动态注册:

效果如下图:

这里就不演示点击按钮布局的实现了,MainActivity.java中实现代码如下:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.Gravity;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
  DynamicReceiver dynamicReceiver;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //实例化IntentFilter对象
    IntentFilter filter = new IntentFilter();
    filter.addAction("panhouye");
    dynamicReceiver = new DynamicReceiver();
    //注册广播接收
    registerReceiver(dynamicReceiver,filter);
  }
  //按钮点击事件
  public void send2(View v){
    Intent intent = new Intent();
    intent.setAction("panhouye");
    intent.putExtra("sele","潘侯爷");
    sendBroadcast(intent);
  }
  /*动态注册需在Acticity生命周期onPause通过
   *unregisterReceiver()方法移除广播接收器,
   * 优化内存空间,避免内存溢出
   */
  @Override
  protected void onPause() {
    super.onPause();
    unregisterReceiver(new MyReceiver());
  }
  //通过继承 BroadcastReceiver建立动态广播接收器
  class DynamicReceiver extends BroadcastReceiver{
    @Override
    public void onReceive(Context context, Intent intent) {
      //通过土司验证接收到广播
      Toast t = Toast.makeText(context,"动态广播:"+ intent.getStringExtra("sele"), Toast.LENGTH_SHORT);
      t.setGravity(Gravity.TOP,0,0);//方便录屏,将土司设置在屏幕顶端
      t.show();
    }
  }
}

建立方法代码中做了详细注释,有不明白的地方请留言讨论。

静态注册:

效果如下:

静态注册建立第一步,新建BroadcastReceiver,见下图:

通过以上步骤,生成MyReceiver.java文件:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.view.Gravity;
import android.widget.Toast;

public class MyReceiver extends BroadcastReceiver {
  public MyReceiver() {
  }
  @Override
  public void onReceive(Context context, Intent intent) {
    Toast t = Toast.makeText(context,"静态广播:"+intent.getStringExtra("info"), Toast.LENGTH_SHORT);
    t.setGravity(Gravity.TOP,0,0);
    t.show();
  }
}

生成MyReceiver.java的同时,修改AndroidMainfest.xml配置文件中的代码:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.example.administrator.day19">
<uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
  <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配置文件
    <receiver
      android:name=".MyReceiver"
      android:enabled="true"
      android:exported="true">
      <intent-filter>
      //自定义Action
        <action android:name="MLY" />
      </intent-filter>
    </receiver>
  </application>
</manifest>

最后在MainActivity.java文件中添加按钮点击事件,如下:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.Gravity;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
  DynamicReceiver dynamicReceiver;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
  }
  //静态广播点击
  public void send(View v){
    Intent intent = new Intent();
    intent.setAction("MLY");
    intent.putExtra("info","panhouye");
    sendBroadcast(intent);
  }
}

至此,两种注册方式的实现代码演示完毕,欢迎探讨。

第二步:为方便后续分析,这里插入BroadcastReceiver的两种常用类型

(1)Normalbroadcasts:默认广播

发送一个默认广播使用Context.sendBroadcast()方法,普通广播对于多个接收者来说是完全异步的,通常每个接收者都无需等待即可以接收到广播,接收者相互之间不会有影响。对于这种广播,接收者无法终止广播,即无法阻止其他接收者的接收动作。

(2)orderedbroadcasts:有序广播

发送一个有序广播使用Context.sendorderedBroadcast()方法,有序广播比较特殊,它每次只发送到优先级较高的接收者那里,然后由优先级高的接受者再传播到优先级低的接收者那里,优先级高的接收者有能力终止这个广播。

发送有序广播:sendorderedBroadCast()

在注册广播中的<intent-filter>中使用android:priority属性。这个属性的范围在-1000到1000,数值越大,优先级越高。在广播接收器中使用setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接收者那里,这样优先级低的接收者可以用getResuttExtras获取到最新的经过处理的信息集合。使用sendorderedBroadcast方法发送有序广播时,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null则表示接收者若要接收此广播,需声明指定权限。这样做是从安全角度考虑的,例如系统的短信就是有序广播的形式,一个应用可能是具有拦截垃圾短信的功能,当短信到来时它可以先接受到短信广播,必要时终止广播传递,这样的软件就必须声明接收短信的权限。

第三步:在默认广播下两种注册方式的比较

(1)两种注册方式均不设置优先级

这里将动态与静态两种注册的广播触发集中在一个按钮上,显示效果如下(未设置优先级的情况下,先动态后静态):

这里同样不演示按钮布局文件,以及静态注册涉及AndroidMainfest.xml和MyReceiver.java文件。直接展示MainActicity.java的实现代码:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.Gravity;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
  DynamicReceiver dynamicReceiver;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    IntentFilter filter = new IntentFilter();
    filter.addAction("panhouye");
    dynamicReceiver = new DynamicReceiver();
    registerReceiver(dynamicReceiver,filter);
  }
  //静态广播点击
  public void send(View v){
    Intent intent = new Intent();
    //设置与动态相同的Action,方便同时触发静态与动态
    intent.setAction("panhouye");
    intent.putExtra("info","潘侯爷");
    sendBroadcast(intent);//默认广播
  }
  @Override
  protected void onPause() {
    super.onPause();
    unregisterReceiver(new MyReceiver());
  }
  class DynamicReceiver extends BroadcastReceiver{
    @Override
    public void onReceive(Context context, Intent intent) {
      Toast t = Toast.makeText(context,"动态广播:"+ intent.getStringExtra("info"), Toast.LENGTH_SHORT);
      t.setGravity(Gravity.TOP,0,0);
      t.show();
    }
  }
}

(2)将动态优先级设置为最低-1000,静态优先级设置为最高1000

显示效果如下(动态仍先于静态被接收到):

MainActivity中动态优先级设置如下:

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    IntentFilter filter = new IntentFilter();
    filter.addAction("panhouye");
    filter.setPriority(-1000);//设置动态优先级
    dynamicReceiver = new DynamicReceiver();
    registerReceiver(dynamicReceiver,filter);
  }

AndroidMainfest.xml中静态优先级设置如下:

<receiver
  android:name=".MyReceiver"
  android:enabled="true"
  android:exported="true">
  //设置静态优先级
  <intent-filter android:priority="1000">
     <action android:name="panhouye" />
  </intent-filter>
</receiver>

第四步:在有序广播下两种注册方式比较

静态广播1(优先级为200),静态广播2(优先级为300),静态广播3(优先级为400),静态广播优先级为(-100),动态广播优先级为0。显示效果如下:

出现顺序由优先级决定,由高到低分别为静态3-静态2-静态1-动-静态。(这里参照前文代码)

第五步:接受打电话的广播,比较程序运行中与结束运行时,两种注册方式的比较

本次比较采用比对Log的方式对两种注册方式进行比较,在MainActivity.java中会插入Activity全部生命周期用于检测Log分析。

AndroidMainfest.xml配置文件代码如下:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.example.administrator.test19">
  //添加拨打电话权限
<uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
  <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=".StaticReceiver"
      android:enabled="true"
      android:exported="true">
      <intent-filter>
        //设置打电话对应的action
        <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
      </intent-filter>
    </receiver>
  </application>
</manifest>

MainActivity.java中实现代码:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;

public class MainActivity extends AppCompatActivity {
  DynamicReceiver dynamicReceiver;//声明动态注册广播接收
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    IntentFilter filter = new IntentFilter();
    filter.addAction("android.intent.action.NEW_OUTGOING_CALL");
    dynamicReceiver = new DynamicReceiver();
    registerReceiver(dynamicReceiver,filter);
    Log.i("Tag","Activity-onCreate");
  }
  @Override
  protected void onStart() {
    super.onStart();
    Log.i("Tag","Activity-onStart");
  }
  @Override
  protected void onResume() {
    super.onResume();
    Log.i("Tag","Activity-onResume");
  }
  @Override
  protected void onPause() {
    super.onPause();
    Log.i("Tag","Activity-onPause");
  }
  @Override
  protected void onStop() {
    super.onPause();
    Log.i("Tag","Activity-onStop");
  }
  @Override
  protected void onDestroy() {
    super.onDestroy();
    Log.i("Tag","Activity-onDestroy");
    unregisterReceiver(dynamicReceiver);
  }
  class DynamicReceiver extends BroadcastReceiver{
    @Override
    public void onReceive(Context context, Intent intent) {
      Log.i("Tag","动态注册广播接收到您正在拨打电话"+getResultData());
    }
  }
}

StaticReceiver.java中实现代码:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
public class StaticReceiver extends BroadcastReceiver {
  public StaticReceiver() {
  }
  @Override
  public void onReceive(Context context, Intent intent) {
    Log.i("Tag","静态注册广播接收到您正在拨打电话"+getResultData());
  }
}

(1)在未退出Activity时,拨打电话,Log如下:

由Log可知在未退出Activity是,两种方式均可接受到广播。

(2)在退出Activity时,拨打电话,Log如下:

在退出程序(Activity)时,只有静态注册方式可以接受到广播。

第六步:总结两种注册方式特点

广播接收器注册一共有两种形式:静态注册和动态注册.

两者及其接收广播的区别:

(1)动态注册广播不是常驻型广播,也就是说广播跟随Activity的生命周期。注意在Activity结束前,移除广播接收器。

静态注册是常驻型,也就是说当应用程序关闭后,如果有信息广播来,程序也会被系统调用自动运行。

(2)当广播为有序广播时:优先级高的先接收(不分静态和动态)。同优先级的广播接收器,动态优先于静态

(3)同优先级的同类广播接收器,静态:先扫描的优先于后扫描的,动态:先注册的优先于后注册的。

(4)当广播为默认广播时:无视优先级,动态广播接收器优先于静态广播接收器。同优先级的同类广播接收器,静态:先扫描的优先于后扫描的,动态:先注册的优先于后册的。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android实现Activity、Service与Broadcaster三大组件之间互相调用的方法详解

    本文实例讲述了Android实现Activity.Service与Broadcaster三大组件之间互相调用的方法.分享给大家供大家参考,具体如下: 我们研究两个问题, 1.Service如何通过Broadcaster更改activity的一个TextView. (研究这个问题,考虑到Service从服务器端获得消息之后,将msg返回给activity) 2.Activity如何通过Binder调用Service的一个方法. (研究这个问题,考虑到与服务器端交互的动作,打包至Service,Ac

  • Android开发之BroadcastReceiver用法实例分析

    本文实例讲述了Android开发中BroadcastReceiver用法.分享给大家供大家参考.具体分析如下: 在Android系统中,广播(Broadcast)是在组件之间传播数据(Intent)的一种机制. Braodcast Receiver顾名思义就是广播接收器,它和事件处理机制类似,但是事件处理机制是程序组件级别的(比如:按钮的单击事件),而广播事件处理机制是系统级别的.我们可以用Intent来启动一个组件,也可以用sendBroadcast()方法发起一个系统级别的事件广播来传递消息

  • Android查看电池电量的方法(基于BroadcastReceiver)

    本文实例讲述了Android查看电池电量的方法.分享给大家供大家参考,具体如下: 程序如下: import android.app.Activity; import android.app.Dialog; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; impor

  • Android编程四大组件之BroadcastReceiver(广播接收者)用法实例

    本文实例讲述了Android编程四大组件之BroadcastReceiver(广播接收者)用法.分享给大家供大家参考,具体如下: 这里介绍如何创建广播.如何发送一个无序广播和有序广播.以及监听短信以及监听呼出电话(当我们发短信和打电话,系统就会发一个广播,我们可以拦截此广播进行监听短信以及监听呼出电话). 定义广播接收者 1.定义类继承BroadcastReceiver,重写onReceive方法 2.当接收到匹配广播之后就会执行onReceive方法 3.清单文件中声明<receiver>,

  • Android提高之BroadcastReceiver实例详解

    前面几篇文章分别讨论了Activity和Service,本文就来讨论BroastcastReceiver,Broastcast是应用程序间通信的手段.BroastcastReceiver也是跟Intent紧密相连的,动态/静态注册了BroastcastReceiver之后,使用sendBroadcast把Intent发送之后,系统会自动把符合条件的BroastcastReceiver启动,这和嵌入式系统的中断类似. 本文所示实例代码主要演示了如何静态/动态注册BroastcastReceiver

  • Android基础知识之broadcast广播详解

    Android中的广播用的太多了,今天稍微总结一下. 按注册方式分为两种: 1.静态注册广播: 静态注册广播就是在androidManifest.xml文件中注册广播,假设我们要实现这样一个效果,在一个activity上点击按钮,发送一条广播,这条广播弹出一个toast,显示"静态"二字. 先看看广播接受者: public class MyBroadcast extends BroadcastReceiver { @Override public void onReceive(Cont

  • Android使用广播(BroadCast)实现强制下线的方法

    本文实例讲述了Android使用广播(BroadCast)实现强制下线的方法.分享给大家供大家参考,具体如下: 使用广播简单模拟强制下线功能. 1.新建活动管理类 新建Activity活动管理类,方便管理所有活动. package com.example.broadcastpractice; import java.util.ArrayList; import java.util.List; import android.app.Activity; /** * 所有活动管理器 * @author

  • Android中BroadcastReceiver(异步接收广播Intent)的使用

    Broadcast Receiver简介 Broadcast Receiver是Android的五大组件之一,使用频率也很高. 用于异步接收广播Intent,广播Intent的发送是通过调用Context.sendBroadcast().广播接收者(BroadcastReceiver)用于异步接收广播Intent,广播Intent的发送是通过调用Context.sendBroadcast().Context.sendOrderedBroadcast()或者Context.sendStickyBr

  • Android中的广播(BroadCast)详细介绍

    什么是广播 在Android中,Broadcast是一种广泛运用的在应用程序之间传输信息的机制.我们拿广播电台来做个比方.我们平常使用收音机收音是这样的:许许多多不同的广播电台通过特定的频率来发送他们的内容,而我们用户只需要将频率调成和广播电台的一样就可以收听他们的内容了.Android中的广播机制就和这个差不多的道理. 电台发送的内容是语音,而在Android中我们要发送的广播内容是一个Intent.这个Intent中可以携带我们要传送的数据. 电台通过大功率的发射器发送内容,而在Androi

  • Android学习之Broadcast的简单使用

    本文实例为大家分享了Android学习之Broadcast的使用方法,供大家参考,具体内容如下 实现开机启动提示网络的广播 package com.example.luobo.broadcasttest; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; impor

随机推荐