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. 20155233 2016-2017-2 《Java程序设计》第6周学习总结

    20155233 2016-2017-2 <Java程序设计>第6周学习总结 学习目标 理解流与IO 理解InputStream/OutPutStream的继承架构 理解Reader/Wr ...

  2. prim算法记录路径

    题目链接:https://vjudge.net/contest/66965#problem/H 代码: #include<iostream> #include<string> ...

  3. Spring Boot1.5X升级到2.0

    配置文件 大量的Servlet专属的server.* properties被移到了server.servlet下 拦截器 public class MyWebMvcConfigurerAdapter ...

  4. AtCoder ARC 090 E / AtCoder 3883: Avoiding Collision

    题目传送门:ARC090E. 题意简述: 给定一张有 \(N\) 个点 \(M\) 条边的无向图.每条边有相应的边权,边权是正整数. 小 A 要从结点 \(S\) 走到结点 \(T\) ,而小 B 则 ...

  5. 【bzoj题解】题解传送门

    如题,题解传送门: 1001 1008 1012

  6. Gitlab权限管理

    使用管理员登陆gitlab(版本为8.9)创建一个组 给用户授权 创建新用户 再创建两个dev1和dev2 然后再到项目界面授权给pm授权master 创建库(事先先建一个java组) 设置权限 创建 ...

  7. Java 并发--线程创建

    随着处理器的多核化,为提高处理器的资源利用率应用程序的并发变应运而生了.现在的操作系统是多任务操作系统,多线程是实现多任务的一种方式. 进程是指一个内存中运行的应用程序,每个进程都有自己独立的内存空间 ...

  8. Android 浏览器启动应用程序

    点击浏览器中的URL链接,启动特定的App. 首先做成HTML的页面,页面内容格式如下: <a href="[scheme]://[host]/[path]?[query]" ...

  9. Codeforces 988F Rain and Umbrellas(DP)

    题目链接:http://codeforces.com/contest/988/problem/F 题目大意: 有三个整数a,n,m,a是终点坐标,给出n个范围(l,r)表示这块区域下雨,m把伞(p,w ...

  10. ZOJ 3469 Food Delivery(区间DP好题)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=4255 题目大意:在x轴上有n个客人,每个客人每分钟增加的愤怒值不同. ...