Android 广播(broadcast)

饮水思源 本文章内容学习和总结自

郭霖大神:《Android第一行代码》

Overview

就像我们的学校里的喇叭一样,是用来通知的。而Android中的广播,要更加的灵活。

广播分类

  • 无序广播:是一种完全异步的执行的广播,广播发出以后,所有的广播接收器几乎都会在同一时间接收到这条广播,没有先后的顺序,效率相对较高,并且无法截断。
  • 有序广播: 是一种同步执行的广播,在广播发出以后,同一时刻只会有一个广播接收器能够收到这条广播消息,该广播接收器的逻辑执行完毕后,广播才会继续传递。广播是有先后顺序的,优先级较高的广播接收者会优先接收到消息,可以经广播进行截断或者向广播中添加新的内容。

Note:

  • 广播接收者,是四大组件中特殊的一个,他可以在清单文件中静态注册,也可以通过代码动态注册。
  • 如果是动态注册的时候,必须在 onDestroy() 生命周期方法中取消注册,否则就会造成内存泄漏。
  • 无序广播无法在,广播传递的过程中截止广播和添加数据。
  • 有序广播可以在,广播传递的过程中添加数据或者截断广播
    • 添加数据使用,this.setResultData(); 等方法.
    • 截断广播使用, this.abortBroadcast();
  • 有序广播通过设置广播接收者的优先级我们可以来确定,哪个广播接收者可以优先接收到广播。
  • 系统的广播
    • 当一个系统的广播,是频发发出的,那么该广播只能通过动态注册的方式来接收,比如 屏幕开关的广播

为什么有序广播的优先级是-1000至1000

接收系统的广播

Android 中内置了许多的系统的广播,我们可以通过监听系统的广播来得到各种系统的状态信息,手机开机发出广播,网络变化发出广播等等。

动态注册广播接受者

Demo: 动态注册广播接收者,接收网络发生变法的广播

package com.example.it.studybroadcast;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.Toast; public class HomeActivity extends AppCompatActivity { private NetWorkReceiver receiver; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_home); //创建一个意图过滤器
IntentFilter filter = new IntentFilter();
//接收网络改变的action
filter.addAction(ConnectivityManager.CONNECTIVITY_ACTION); receiver = new NetWorkReceiver();
//注册广播
this.registerReceiver(this.receiver, filter);
} /*
* 取消注册
* */
@Override
protected void onDestroy() {
super.onDestroy();
this.unregisterReceiver(this.receiver);
} public class NetWorkReceiver extends BroadcastReceiver { @Override
public void onReceive(Context context, Intent intent) { ConnectivityManager connectivityManager = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE); //如果当前有默认的网络就返回NetworkInfo 否则就返回 null
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
//因为可能是null 所以要先判断是否为空
if (networkInfo != null && networkInfo.isAvailable()) {
if (networkInfo.getType() == ConnectivityManager.TYPE_WIFI) {
Toast.makeText(context, "Wifi", Toast.LENGTH_SHORT).show(); } else if (networkInfo.getType() == ConnectivityManager.TYPE_MOBILE) {
Toast.makeText(context, "流量", Toast.LENGTH_SHORT).show();
}
} else {
Toast.makeText(context, "世界上最遥远的距离就是没有网", Toast.LENGTH_SHORT).show();
}
}
}
}

静态注册广播接收者

Demo:实现开机启动

清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.it.studybroadcast"> <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /> <application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".HomeActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".RootComplatedActivity"></activity>
<!--注册我们的开机广播-->
<receiver
android:name=".MyBootComplatedReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
</application> </manifest>

Java 代码

public class MyBootComplatedReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Intent gotoIntent = new Intent(context, RootComplatedActivity.class);
context.startActivity(gotoIntent);
}
}

自定义广播

无序广播

发送一个无序广播

public void sendOrdered(View view) {
Intent intent = new Intent();
//放入我们的数据
intent.putExtra("MyKey", "我是自定义发送的广播");
//放我我们Action,用来区分其他广播,注意Acting不要和系统的广播重复
intent.setAction("com.example.it.studybroadcast.CustomBroadcast");
this.sendBroadcast(intent);
}

定义广播接收者接收我们的无序广播

清单文件中配置一下广播接收者

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.it.studybroadcast">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
------
<receiver
android:name=".MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.example.it.studybroadcast.CustomBroadcast" />
</intent-filter>
</receiver>
-----
</application> </manifest>

Java 代码

public class MyReceiver extends BroadcastReceiver {
public MyReceiver() {
} @Override
public void onReceive(Context context, Intent intent) {
String value = intent.getStringExtra("MyKey");
Toast.makeText(context, value, Toast.LENGTH_SHORT).show();
}
}
有序广播

清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.it.studybroadcast">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
------
<receiver
android:name=".MyOrderedReceiverOne"
android:enabled="true"
android:exported="true">
<!--设置我们的优先级 优先级的范围是 -1000到1000-->
<intent-filter android:priority="1000">
<action android:name="com.example.it.studybroadcast.MyOrderedBroadcast" />
</intent-filter>
</receiver>
<receiver
android:name=".MyOrderedBroadcastTwo"
android:enabled="true"
android:exported="true">
<intent-filter android:priority="999">
<action android:name="com.example.it.studybroadcast.MyOrderedBroadcast" />
</intent-filter>
</receiver>
-------------
</application>
</manifest>

Java 代码

/*
* 发送一个有序广播
* */
public void sendOrderedBroadcast(View view) {
Intent intent = new Intent();
intent.setAction("com.example.it.studybroadcast.MyOrderedBroadcast");
//放入一些数据
intent.putExtra("MyKey", "MyKeyVlaue");
/*
* 参数:
* intent: 我们发送的意图
* receiverPermission: 广播接收者必须有的权限
* */
this.sendOrderedBroadcast(intent, null);
} /*
广播接收者
*/
public class MyOrderedReceiverOne extends BroadcastReceiver {
public MyOrderedReceiverOne() {
} @Override
public void onReceive(Context context, Intent intent) {
String value = intent.getStringExtra("MyKey");
Toast.makeText(context, value, Toast.LENGTH_SHORT).show(); //添加数据
this.setResultData("我是添加的内容!");
}
} public class MyOrderedBroadcastTwo extends BroadcastReceiver {
public MyOrderedBroadcastTwo() {
} @Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, getResultData(), Toast.LENGTH_SHORT).show();
//截断我们的广播
this.abortBroadcast();
}
}

源码下载

https://git.oschina.net/ShareKnowledge/StudyBroadcast

Android 中的广播(Broadcast)的更多相关文章

  1. Android中的广播Broadcast详解

    今天来看一下Android中的广播机制,我们知道广播Broadcast是Android中的四大组件之一,可见他的重要性了,当然它的用途也很大的,比如一些系统的广播:电量低.开机.锁屏等一些操作都会发送 ...

  2. Android中的广播

    Android中的广播 广播接受器,可以比喻成收音机.而广播则可以看成电台. Android系统内部相当于已经有一个电台 定义了好多的广播事件,比如外拨电话 短信到来 sd卡状态 电池电量变化... ...

  3. Android 中的广播机制

    Android 中的广播机制 Android 中的广播,按照广播响应范围,可以分为应用内广播和全局广播.按照广播的接收方式,可以分为标准广播和有序广播. 广播的分类 响应范围 应用内广播:此类广播只能 ...

  4. android中的广播接收实现总结

    1 首先根据广播应用内接收和应用外接收,分两个类进行管理[1]  LocalBroadcastManager,应用内广播管理类[2]  BroadcastManager  广播管理类(部分应用内,应用 ...

  5. Android中的广播基本实现及回调方法的理解

    在Android中broadcast这一节的内容其实不算多主要是牵扯到一个broadcastreceiver类,这个类是一个抽象类,下面有一个抽象方法onreceiver(),可以再我们收到网络状态变 ...

  6. 我的Android进阶之旅------>Android中MediaButtonReceiver广播监听器的机制分析

    今天看公司的一段关于MediaButtonReceiver的代码看的比较混乱,幸好看了下面的这篇文章,才能茅塞顿开的理解好代码.在此转载下来,以备以后理解,希望都到这篇文章的人也能够有所帮助. 本文转 ...

  7. Android中使用广播机制退出多个Activity

    谷歌百度一下,Android中退出多个Activity的方法,大家讨论的很多. 在实习的时候,看到公司的项目退出多个Activity,是采用LinkedList方法,毕业设计的时候,也参照了那种方法. ...

  8. Android中本地广播的实现

    其实Android的本地广播并没有什么好讲的,他就是用了一个localbroadcastmanager类来sendbroadcast,以及注册和注销广播,没有什么特点,其中实例该类的时候用了getin ...

  9. Android为什么需要广播Broadcast

       在Android系统中,为什么需要广播机制呢?广播机制,本质上它就是一种组件间的通信方式,如果是两个组件位于不同的进程当中,那么可以用Binder机制来实现,如果两个组件是在同一个进程中,那么它 ...

随机推荐

  1. TIP协议

    1. TIP是什么? CISCO给TIP的定义如下: The TIP protocol specifications describe how to multiplex multiple screen ...

  2. summernote 文本编辑器使用时,选择上传图片、链接、录像时,弹出的对话框被遮挡住

    更多内容推荐微信公众号,欢迎关注: 即问题如下链接内的情况: http://bbs.csdn.net/topics/392004332 这个一般属于CSS中样式出现了问题,可以在点开的时候,F12查看 ...

  3. c# 生成随机N位数字串(每位都不重复)

    /// <summary> /// 生成随机数字窜 /// </summary> /// <param name="Digit">位数</ ...

  4. c语言学习笔记.链表.

    链表: 链表单个节点的数据结构.链表的实现主要依靠结构体和指针. 头指针(head)指向链表的第一个节点,然后第一个节点中的指针指向下一个节点,然后依次指到最后一个节点,这样就构成了一条链表. str ...

  5. 利用rundll32执行程序的函数执行程序

    1.前言 无意间发现hexacorn这个国外大佬,给出了很多通过rundll32执行DLL中的函数执行程序的方法,思路很灵巧. 2.原理 rundll32加载dll 用法: rundll32 < ...

  6. python socket编程和黏包问题

    一.基于TCP的socket tcp是基于链接的,必须先启动服务端,然后再启动客户端去链接服务端,有顺序,不重复,可靠.不会被加上数据边界. server端 import socket sk = so ...

  7. Genymotion上不能安装APK软件的问题

    Genymotion模拟器不能安装APK的原因 官网给出的解释:Genymotion模拟器使用的是x86架构,在第三方市场上的应用有部分不采用x86这么一种架构,所以在编译的时候不通过,报“APP n ...

  8. mariadb/mysql使用Navicat连接报错

    [问题1] 使用Navicat连接服务器的mariadb/mysql时报错 access denied for user root@192.168.xx.xx(using password:yes) ...

  9. MongoDB-MongoDB重装系统后恢复

    重装系统后,把原mongoDB安装目录和原mongoDB的data目录拷贝到新硬盘的D盘上. 恢复的方法如下. 1.D:\Mongodb里放着mongod.cfg和data C:\Users\Admi ...

  10. ZOJ 3537 Cake(凸包+区间DP)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3537 题目大意:给出一些点表示多边形顶点的位置,如果不是凸多边形 ...