• Android 广播 BroadcastReceiver


    Android 系统里定义了各种各样的广播,如电池的使用状态,电话的接收和短信的接收,开机启动都会产生一个广播。当然用户也可以自定义自己的广播。

    既然说到广播,那么必定有一个广播发送者,以及广播接收器。系统广播的发送者为系统,自定义广播当然是用户定义的了。

    我们可以定义一个广播接收器,用来接收我们感兴趣的广播,不论是系统广播还是用户自定义广播。这个广播接收器必须继承至BroadcastReceiver。

    老规矩,先来点基础知识。

    一.基础知识

    1:定义一个广播接收器,继承BroadcastReceiver。

    2:在广播接收器中onReceive接收广播的Action并作出处理。不同的广播对于一组不同的Action。

    3:在manifest.xml文件中注册广播,或者通过registerReceiver方法注册广播,当然unregisterReceiver可以删除广播。

    下面给出系统定义好的广播,即系统广播的Action

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

    二.实战

    假设我们监听短信,并获取短信内容为hello时提示,还有,同时我们监听我们自定义的广播,并作出反应。代码如下

    package com.dongzi;
    
    import android.content.BroadcastReceiver;
    import android.content.Context;
    import android.content.Intent;
    import android.os.Bundle;
    import android.telephony.SmsMessage;
    import android.widget.Toast;
    
    /**
     * 监听SMS信息广播,以及MSG自定义广播
     * @author  
     *
     */
    public class SMSReceiver extends BroadcastReceiver {
          
        //系统广播,自定义广播
        static final String SMS_ACTION="android.provider.Telephony.SMS_RECEIVED";
        static final String MSG_ACTION="com.dongzi.customMsg";
        static final String HELLO="hello";
        static final String BUNDLE="bundle";
        String receiveMsg="";
        @Override
        public void onReceive(Context context, Intent intent) {
            
             SmsMessage[] msg=null;
             Bundle bundle=null;
             //这里监听系统广播
             if(intent.getAction().equals(SMS_ACTION)){
                  bundle=intent.getExtras(); 
                if(bundle != null) {
                Object[] objs=(Object[])bundle.get("puds");
                msg=new SmsMessage[objs.length];
                 for(int i=0;i<objs.length;i++)
                     msg[i]= SmsMessage.createFromPdu((byte[])objs[i]);
         
                    for(int j=0;j<msg.length;j++){
                        String msg_str=msg[j].getMessageBody();
                        if(msg_str.equals(HELLO)){
                            Toast.makeText(context, msg_str, Toast.LENGTH_LONG).show();
                            return;
                        }else{
                            Toast.makeText(context, msg_str, Toast.LENGTH_LONG).show();
                            return;
                        }
                    }
                 }
                
             }//这里监听我们自定义的广播 
             else if(intent.getAction().equals(MSG_ACTION)){
                  bundle=intent.getBundleExtra(BUNDLE);
                  if(bundle != null){
                    String hello=bundle.getString(HELLO);
                    if(hello!=null &&hello.equals(HELLO)){
                        Toast.makeText(context, hello, Toast.LENGTH_LONG).show();
                        return;
                      }
                  }
             }
             
        }
    
    }

    然后我们可以在代码中注册,或者在配置文件中注册

    //IntentFilter filter=new IntentFilter();
         ////filter.addAction(SMSReceiver.MSG_ACTION);
         //registerReceiver(SMSReceiver.class.newInstance(), filter);
         //unregisterReceiver(receiver)

    <receiver android:name=".SMSReceiver">
                <intent-filter>
                    <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
                    <action android:name="com.dongzi.customMsg"/>
                </intent-filter>
            </receiver>

    他们会拦截凡是符合action的广播

    然后我们再代码中发送广播吧。

    //发送自定义广播
        private void sendCustomBroadcast(){
             Intent intent=new Intent(SMSReceiver.MSG_ACTION);
             Bundle bundle=new Bundle();
             bundle.putString(SMSReceiver.HELLO, "hello");
             intent.putExtra(SMSReceiver.BUNDLE, bundle);
            sendBroadcast(intent);
        }
  • 相关阅读:
    两角和的正切
    积化和差与和差化积
    require.js的简单使用
    HTML、css、javascript、DOM编程
    SignalR长连接的简单用法
    【ESP8266】发送HTTP请求
    记录自己的第一篇博客
    1 为什么搭建.Net core下的云开发框架
    C#线程中LOCK的意义
    ping命令执行过程详解
  • 原文地址:https://www.cnblogs.com/yishujun/p/3591226.html
Copyright © 2020-2023  润新知