Androi四大核心组件之BroadcastReceiver

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.youkuaiyun.com/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.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;
//设备墙纸已改变时发出的广播
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

王家视频教程图书馆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值