Androi四大核心组件之BroadcastReceiver

来源:互联网 发布:科比场均数据每年 编辑:程序博客网 时间:2024/05/22 15:37

Android广播机制简介

在Android中,有一些操作完成以后,会发送广播,比如说发出一条短信,或打出一个电话,如果某个程序接收了这个广播,就会做相应的处理。这个广播跟我们传统意义中的电台广播有些相似之处。之所以叫做广播,就是因为它只负责“说”而不管你“听不听”,也就是不管你接收方如何处理。另外,广播可以被不只一个应用程序所接收,当然也可能不被任何应用程序所接收。
广播机制最大的特点就是发送方并不关心接收方是否接到数据,也不关心接收方是如何处理数据的。
Android中广播的是操作系统中产生的各种各样的事件。例如,收到一条短信就会产生一个收到短信息的事件。而Android操作系统一旦内部产生了这些事件,就会向所有的广播接收器对象来广播这些事件。

Android广播分为两个方面:
广播发送者和广播接收者,通常情况下,BroadcastReceiver指的就是广播接收者(广播接收器)。广播作为Android组件间的通信方式,可以使用的场景如下:

1.同一app内部的同一组件内的消息通信(单个或多个线程之间);2.同一app内部的不同组件之间的消息通信(单个进程);3.同一app具有多个进程的不同组件之间的消息通信;4.不同app之间的组件之间消息通信;

发送一个默认广播:

MainActivity.java

//发送一个普通广播    public void button1(View v) {        Intent intent=new Intent("com.long");        this.sendBroadcast(intent);    }

Activity1.java

实现广播接收器,你要写一个自己的类继承BroadcastReceiver类,重写onReceive方法,并在此方法中实现核心业务代码

public class Activity1 extends BroadcastReceiver {    //当接收到广播执行的方法    @Override    public void onReceive(Context context, Intent intent) {        // TODO Auto-generated method stub        Toast.makeText(context,"我是一个默认广播",1000).show();    }

最后一步不要忘了注册:

<receiver android:name=".Activity1"            android:enabled="true"            android:exported="true">            <intent-filter >                <action android:name="com.long"/>            </intent-filter>

有序广播
//可以中断广播
//this.abortBroadcast();

MainActivity.java

//发送一个有序广播        public void button2(View v) {            Intent intent=new Intent("com.long.long");            this.sendOrderedBroadcast(intent, null);        }

AndroidManifest.xml

<receiver android:name=".Activity2"            android:enabled="true"            android:exported="true">            <intent-filter >                <action android:name="com.long.long"/>            </intent-filter>        </receiver>        <receiver android:name=".Activity3"            android:enabled="true"            android:exported="true">            <intent-filter >                <action android:name="com.long.long"/>            </intent-filter>        </receiver>

粘性广播

发送粘性广播需要权限(这里的权限是保存信息的权限和由系统发送未处理的广播的权限)

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

粘性广播请参考此原理图:
这里写图片描述


代码注册广播接收器

public class MainActivity extends Activity {    Activity1 activity1=new Activity1();    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);    }    //在该方法中进行广播注册    @Override    protected void onResume() {        // TODO Auto-generated method stub        super.onResume();        IntentFilter filter=new IntentFilter();        filter.addAction("com.long");        registerReceiver(activity1, filter);    }    //在该方法中解除广播注册    @Override    protected void onPause() {        // TODO Auto-generated method stub        super.onPause();        unregisterReceiver(activity1);    }}

接收开机启动广播核心代码

声明开机广播接收权限:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<receiver android:name=".my">            <intent-filter >                <action android:name="android.Intent.ACTION_BOOT_COMPLETED"/>            </intent-filter>        </receiver>

监听网络状态变化——广播接收者

手机的网络状态监听:

  android系统已经有这样一个动作,当手机的网络状态发生改变时,发送一个广播;

  所以我只需要做的步骤是:

 1、添加允许获取网络切换的状态权限:<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />     添加网络连接的权限:<uses-permission android:name="android.permission.INTERNET" />    2、在AndroidMainifest.xml中注册该广播接收者:  <receiver android:name="com.example.newwork.NetworkReceiver" >    <intent-filter>      <!-- 监听的动作类型,这里是监听网络连接的状态  -->      <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />    </intent-filter>  </receiver>

自己的BrodCaseReceiver:

public class NetworkReceiver extends BroadcastReceiver {    @Override    public void onReceive(Context context, Intent intent) {      ConnectivityManager manager = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);      NetworkInfo activeInfo = manager.getActiveNetworkInfo();       //如果无网络连接activeInfo为null      //也可获取网络的类型      if(activeInfo != null){ //网络连接        Toast.makeText(context, "测试:网络连接成功",0).show();      }else { //网络断开        Toast.makeText(context, "测试:网络断开",0).show();      }    }  }

  应用场景:

    qq在断网的时候无论你的在哪个页面 当前刷新都会出现网络断开的提示,可以通过 模板模式 + 观察者模式的设计实现;


监听系统电量变化

当手机电量发生改变时,系统会对外发送Intent的Action为android.intent.action.BATTERY_CHANGED常量的广播;当手机电量过低时,系统会对外发送Intent的Action为android.intent.action.BATTERY_LOW常量的广播。当手机电池从电量不足状态恢复时,系统会对外发送Intent的Action为android.intent.action.BATTERY_OKAY常量的广播。

下面通过一个简单实例来演示:

    package com.home.receiver;      import android.content.BroadcastReceiver;      import android.content.Context;      import android.content.Intent;      import android.os.Bundle;      import android.widget.Toast;      public class BatteryReceiver extends BroadcastReceiver {          @Override          public void onReceive(Context context, Intent intent) {              if (Intent.ACTION_BATTERY_OKAY.equals(intent.getAction())) {                  Toast.makeText(context, "电量已恢复,可以使用!", Toast.LENGTH_LONG).show();              }              if (Intent.ACTION_BATTERY_LOW.equals(intent.getAction())) {                  Toast.makeText(context, "电量过低,请尽快充电!", Toast.LENGTH_LONG).show();              }              if (Intent.ACTION_BATTERY_CHANGED.equals(intent.getAction())) {                  Bundle bundle = intent.getExtras();                  // 获取当前电量                  int current = bundle.getInt("level");                  // 获取总电量                  int total = bundle.getInt("scale");                  StringBuffer sb = new StringBuffer();                  sb.append("当前电量为:" + current * 100 / total + "%" + "  ");                  // 如果当前电量小于总电量的15%                  if (current * 1.0 / total < 0.15) {                      sb.append("电量过低,请尽快充电!");                  } else {                      sb.append("电量足够,请放心使用!");                  }                  Toast.makeText(context, sb.toString(), Toast.LENGTH_LONG).show();              }          }      }  

添加相应权限

 <receiver android:name="com.home.receiver.BatteryReceiver">                <intent-filter>                    <action android:name="android.intent.action.BATTERY_CHANGED" />                    <action android:name="android.intent.action.BATTERY_OKAY"/>                    <action android:name="android.intent.action.BATTERY_LOW"/>                </intent-filter>              </receiver>  

Android中的AlarmManager的使用

1、AlarmManager,顾名思义,就是“提醒”,是Android中常用的一种系统级别的提示服务,在特定的时刻为我们广播一个指定的Intent。简单的说就是我们设定一个时间,然后在该时间到来时,AlarmManager为我们广播一个我们设定的Intent,通常我们使用 PendingIntent,PendingIntent可以理解为Intent的封装包,简单的说就是在Intent上在加个指定的动作。在使用Intent的时候,我们还需要在执行startActivity、startService或sendBroadcast才能使Intent有用。而PendingIntent的话就是将这个动作包含在内了。

AlarmManager详解传送门:
http://blog.csdn.net/wangxingwu_314/article/details/8060312
核心代码

public void button1(View v) {    AlarmManager am=(AlarmManager) getSystemService(Context.ALARM_SERVICE);    //设置触发闹钟的时间    long time=System.currentTimeMillis()+3000;    Intent intent=new Intent(this,alarm.class);    PendingIntent pi=PendingIntent.getBroadcast(this,0,intent,PendingIntent.FLAG_NO_CREATE);    am.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, time, pi);    }

最后给大家提供下我们平常可能会用到的一些系统广播吧:

Intent.ACTION_AIRPLANE_MODE_CHANGED;//关闭或打开飞行模式时的广播<strong>Intent.ACTION_BATTERY_CHANGED;//充电状态,或者电池的电量发生变化//电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册<strong>Intent.ACTION_BATTERY_LOW;//表示电池电量低<strong>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.javaIntent.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;//设备墙纸已改变时发出的广播