Android广播的两种类型:

1.静态广播

2.动态广播

静态注册广播:

Manifeast中的代码块:

<receiver
android:name=".broadcast.MyStaticReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="husanity"/>
</intent-filter>
</receiver>
MyStaticReceiver这个类里面很简单,只做了打印收到的信息:
package com.hxc.supreme.broadcast;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.widget.Toast; import com.hxc.supreme.utils.ToastUtil; public class MyStaticReceiver extends BroadcastReceiver { @Override
public void onReceive(Context context, Intent intent) {
Log.i("MyStaticReceiver", "onReceive: "+intent.getStringExtra("info"));
}
}

然后就是主界面一个点击按钮,布局文件很简单:

package com.hxc.supreme.activity;

import android.content.Intent;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView; import com.hxc.supreme.R; /**
* created by huxc on 2017/9/28.
* func:Android广播详解
* email: hxc242313@qq.com
*/ public class BroadcastActivity extends AppCompatActivity implements View.OnClickListener { private TextView tvStaticBroadcast;
private TextView tvDynamicBroadcast; @Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_broadcast);
tvStaticBroadcast = findViewById(R.id.tv_static_broadcast);
tvDynamicBroadcast = findViewById(R.id.tv_dynamic_broadcast);
tvStaticBroadcast.setOnClickListener(this);
tvDynamicBroadcast.setOnClickListener(this);
} @Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.tv_static_broadcast:
sendBroadcast();
break;
case R.id.tv_dynamic_broadcast:
break;
}
} private void sendBroadcast() {
Intent intent = new Intent();
intent.setAction("husanity");
intent.putExtra("info", "这是一条静态广播");
sendBroadcast(intent);
} }

然后看一下logcat中的日志输出:

02-15 15:33:14.300 15384-15384/com.hxc.supreme I/MyStaticReceiver: onReceive: 这是一条静态广播

然而,在Android8.0及以上的系统中,收不到静态广播!Google了一下发现,谷歌在8.0后为了提高效率,删除了静态注册,防止关闭App后广播还在, 造成内存泄漏, 现在静态注册的广播需要指定包名!!!

意思就是在8.0以后Google官方推荐用动态注册替换静态注册,那我们就指定一下包名看看是否管用:

代码只改动了发送广播的方法:

 private void sendBroadcast() {
Intent intent = new Intent();
intent.setPackage(getPackageName());//Android8.0以上需指定包名
intent.setAction("husanity");
intent.putExtra("info", "这是一条静态广播");
sendBroadcast(intent);
}

看下logcat的输出日志:

02-15 15:46:21.891 11181-11181/com.hxc.supreme I/MyStaticReceiver: onReceive: 这是一条静态广播

事实是证明可以的,但是Google推荐使用动态注册,那我么接下来就看一下动态注册的使用方式。

动态注册广播:

直接上代码:

package com.hxc.supreme.activity;

import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView; import com.hxc.supreme.R; /**
* created by huxc on 2017/9/28.
* func:Android广播详解
* email: hxc242313@qq.com
*/ public class BroadcastActivity extends AppCompatActivity implements View.OnClickListener { private TextView tvStaticBroadcast;
private TextView tvDynamicBroadcast;
private DynamicReceiver dynamicReceiver; @Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_broadcast);
tvStaticBroadcast = findViewById(R.id.tv_static_broadcast);
tvDynamicBroadcast = findViewById(R.id.tv_dynamic_broadcast);
tvStaticBroadcast.setOnClickListener(this);
tvDynamicBroadcast.setOnClickListener(this);
dynamicReceiver = new DynamicReceiver();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("husanity");
registerReceiver(dynamicReceiver, intentFilter);
} @Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.tv_static_broadcast:
sendBroadcast();
break;
case R.id.tv_dynamic_broadcast:
sendDynamicBroadcast();
break;
}
} private void sendBroadcast() {
Intent intent = new Intent();
intent.setPackage(getPackageName());//Android8.0以上需指定包名
intent.setAction("husanity");
intent.putExtra("info", "这是一条静态广播");
sendBroadcast(intent);
} private void sendDynamicBroadcast() {
Intent intent = new Intent();
intent.setAction("husanity");
intent.putExtra("extra", "这是一条动态广播");
sendBroadcast(intent); } class DynamicReceiver extends BroadcastReceiver { @Override
public void onReceive(Context context, Intent intent) {
Log.i("BroadcastActivity", "onReceive: " + intent.getStringExtra("extra")); }
} @Override
protected void onDestroy() {
super.onDestroy();
//onDestroy中需要反注册,用来优化内存空间避免内存泄漏
unregisterReceiver(dynamicReceiver);
}
}

看下log cat打印日志:

02-15 16:08:25.559 11597-11597/com.hxc.supreme I/BroadcastActivity: onReceive: 这是一条动态广播

可以看到结果也是能够收到广播的,那么既然两种都可以,那静态注册和动态注册到底有什么区别呢?

静态注册和动态注册的区别:

通过一个拨打电话然后监听收到的打电话的广播来看看两者的区别:

manifest修改:

<receiver
android:name=".broadcast.MyStaticReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<!--<action android:name="husanity"/>-->
<!--设置打电话对应的action-->
<action android:name="android.intent.action.NEW_OUTGOING_CALL" />
</intent-filter>
</receiver>

Java代码中动态广播修改:(Android6.0之后的获取通讯录权限需要动态获取)

<receiver
android:name=".broadcast.MyStaticReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<!--<action android:name="husanity"/>-->
<!--设置打电话对应的action-->
<action android:name="android.intent.action.NEW_OUTGOING_CALL" />
</intent-filter> </receiver>

静态广播修改:

package com.hxc.supreme.broadcast;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.widget.Toast; import com.hxc.supreme.utils.ToastUtil; public class MyStaticReceiver extends BroadcastReceiver { @Override
public void onReceive(Context context, Intent intent) {
Log.i("MyStaticReceiver", "静态广播接收到您正在拨打电话 " + getResultData());
}
}

当我们直接从BroadcastActivity这个界面回到手机界面打电话时,看一下logcat的输出:

02-15 16:30:46.162 4744-4744/com.hxc.supreme I/BroadcastActivity: 动态广播接收到您正在拨打电话 10001
02-15 16:30:46.200 4744-4744/com.hxc.supreme I/MyStaticReceiver: 静态广播接收到您正在拨打电话 10001

然后从BroadcastActivity界面先返回上级界面然后再回到手机界面拨打电话,此时的logcat输出:

02-15 16:40:04.007 4744-4744/com.hxc.supreme I/com.hxc.supreme.activity.BroadcastActivity: onDestroy:
02-15 16:40:27.189 4744-4744/com.hxc.supreme I/MyStaticReceiver: 静态广播接收到您正在拨打电话 10001

这是只能收到静态注册的广播,无法收到动态注册的广播了。

所以,静态广播是常驻型广播,不随界面的销毁而销毁,只要程序没被杀死,收到了对应的广播通知时就会调用程序;

而动态注册广播跟随界面的生命周期,不要忘了在页面销毁之前释放广播。

有序广播和普通广播:

普通广播:

通过Context.sendBroadcast()方法来发送,它是完全异步的。所有的receivers(接收器)的执行顺序不确定,因此所有的receivers(接收器)接收broadcast的顺序不确定。
 
有序广播:
通过Context.sendOrderedBroadcast来发送,所有的receiver依次执行。BroadcastReceiver可以使用setResult系列函数来结果传给下一个BroadcastReceiver,通过getResult系列函数来取得上个BroadcastReceiver返回的结果,并可以abort系列函数来让系统丢弃该广播,使用该广播不再传送到别的BroadcastReceiver。可以通过在intent-filter中设置android:priority属性来设置receiver的优先级,优先级相同的receiver其执行顺序不确定。
 
举例:
短信拦截:你可以设置自己的app的广播接收器的级别高于系统原来的级别,并且收到后中断往下传播,这样的话收件箱里面就不会收到短信了。

Android 四大组件之broadcast的理解的更多相关文章

  1. android四大组件之Broadcast

    广播的概念 现实中:我们常常使用电台通过发送广播发布消息,买个收音机,就能收听 Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件.Androi ...

  2. [Android基础]Android四大组件之BroadCast

    BroadCast的定义: 广播是一种订阅--通知 事件,广播接收者向Android系统 register (订阅广播),广播发送者向Adnroid系统 sendBroadCast(发送广播),然后A ...

  3. 【Android开发日记】之入门篇(六)——Android四大组件之Broadcast Receiver

    广播接受者是作为系统的监听者存在着的,它可以监听系统或系统中其他应用发生的事件来做出响应.如设备开机时,应用要检查数据的变化状况,此时就可以通过广播来把消息通知给用户.又如网络状态改变时,电量变化时都 ...

  4. Android四大组件之BroadCast(续)

    1.哪一个方法可以发送广播? activity.sendbroadcast or context.sentbroadcast or service.sendbroadcast 2.创建广播接受程序必须 ...

  5. Android 四大组件之service与Broadcast

    Android 四大组件之一:service: Service有五个生命周期:onCreat,onStartCommand, onBind,onUnbind, onDestroy 主要有绑定和非绑定两 ...

  6. Android四大组件之一“广播”

    前言 Android四大组件重要性已经不言而喻了,今天谈谈的是Android中的广播机制.在我们上学的时候,每个班级的教室里都会装有一个喇叭,这些喇叭都是接入到学校的广播室的,一旦有什么重要的通知,就 ...

  7. 第九章:四大组件之Broadcast Receiver

    第九章:四大组件之Broadcast Receiver   一.广播的功能和特征 广播的生命周期很短,经过调用对象-->实现onReceive-->结束,整个过程就结束了.从实现的复杂度和 ...

  8. Android 四大组件之“ BroadcastReceiver ”

    前言 Android四大组件重要性已经不言而喻了,今天谈谈的是Android中的广播机制.在我们上学的时候,每个班级的教室里都会装有一个喇叭,这些喇叭都是接入到学校的广播室的,一旦有什么重要的通知,就 ...

  9. Android四大组件与进程启动的关系(转)

    一. 概述 Android系统将进程做得很友好的封装,对于上层app开发者来说进程几乎是透明的. 了解Android的朋友,一定知道Android四大组件,但对于进程可能会相对较陌生. 一个进程里面可 ...

随机推荐

  1. android发布新版忘记keystore(jks)密码终极解决方案

    android app签名是使用的keystore文件/jks文件,如果是eclipse是keystore,android studio则是jks,如果忘记了的话很悲催: 1.找到密码 2.改应用的包 ...

  2. Spring Data Redis 详解及实战一文搞定

    SDR - Spring Data Redis的简称. Spring Data Redis提供了从Spring应用程序轻松配置和访问Redis的功能.它提供了与商店互动的低级别和高级别抽象,使用户免受 ...

  3. nRF52832 SDK15.3.0 基于ble_app_uart demo FreeRTOS移植

    参考资料:https://blog.csdn.net/u010860832/article/details/86235993 这里把移植经验记录下来,供有需要的同学参考,有不对的地方也请大家批评指正. ...

  4. VueJs(7)---计算属性和侦听器

    计算属性和侦听器 一. 概述 计算属性 模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的.在模板中放入太多的逻辑会让模板过重且难以维护.例如: <div id="exampl ...

  5. logstash收集syslog日志

    logstash收集syslog日志注意:生产用syslog收集日志!!! 编写logstash配置文件 #首先我用rubydebug测试数据 [root@elk-node1 conf.d]# cat ...

  6. 详解CSS的Flex布局

    本文由云+社区发表 Flex是Flexible Box 的缩写,意为"弹性布局",是CSS3的一种布局模式.通过Flex布局,可以很优雅地解决很多CSS布局的问题.下面会分别介绍容 ...

  7. 项目实战4—HAProxy实现高级负载均衡实战和ACL控制

     haproxy实现高级负载均衡实战 环境:随着公司业务的发展,公司负载均衡服务已经实现四层负载均衡,但业务的复杂程度提升,公司要求把mobile手机站点作为单独的服务提供,不在和pc站点一起提供服务 ...

  8. python三大神器之fabric(2.0新特性)

    fabric经常出现在自动化运维领域,批量处理一些运维工作.fabric是在paramiko之上又封装了一层,操作起来更加简单易用. 本来只是想写个博客记录一下,然后发现之前写的代码不能运行了,报以下 ...

  9. vue的路由映射问题

    遇到的问题 今天在项目中遇到了一个问题,明明在Router文件夹下的路由js映射文件中,配置好了,如下: // 生日贺卡 { path: 'birthdayRemind', component: lo ...

  10. .NET 中的序列化 & 反序列化

    序列化:将对象的状态信息及类型信息,转换为一种易于传输或存储形式(流,即字节序列)的过程. 下图为序列化过程图示,图片来自微软官方文档: 反序列化:与序列化相反,将流转换为对象的过程. 常用的有二进制 ...