【Android】安卓四大组件之广播知识总结

文章目录

  • 动态注册
    • 使用BroadcastReceiver监听Intent广播
    • 注册Broadcast Receiver
  • 静态注册
  • 自定义广播
    • 标准广播
      • 发送广播
      • 定义广播接收器
      • 注册广播接收器
    • 有序广播
      • 修改发送方法
      • 定义第二个广播接收器
      • 注册广播接收器
      • 广播截断
  • 使用本地广播
  • 实践-强制下线
    • 使用ActivityCollector管理所有活动
    • 登录界面
    • 主界面
    • BaseActivity

我们知道Intent相当于快递员,在安卓的活动,广播这些组件中传输数据,进行通信。
这篇博客就来学习使用Intent通过sendBroadcast方法在组件之间广播消息

动态注册

使用BroadcastReceiver监听Intent广播

用于监听要监听的广播,必须进行注册。

方法:新建一个内部类继承自BroadcastReceiver,重写父类onReceive方法

这种方法称为动态注册,只有在应用组件运行时才能响应广播Intent

public class MainActivity extends AppCompatActivity {//...// 内部类:广播接收器,用于处理网络连接变化的广播class NetworkChangeReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 获取 ConnectivityManager 实例ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);// 获取当前活动的网络信息NetworkInfo activeNetworkInfo = connectivityManager.getActiveNetworkInfo();// 检查网络是否可用if (activeNetworkInfo != null && activeNetworkInfo.isAvailable()) {// 网络可用时,显示提示Toast.makeText(context, "Network is available", Toast.LENGTH_SHORT).show();} else {// 网络不可用时,显示提示Toast.makeText(context, "Network is unavailable", Toast.LENGTH_SHORT).show();}}}
}

注册Broadcast Receiver

一般在onStart注册,onStop解注册

public class MainActivity extends AppCompatActivity {// 声明 IntentFilter 和 BroadcastReceiverprivate IntentFilter intentFilter;private NetworkChangeReceiver networkChangeReceiver;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 初始化 IntentFilter 并添加网络连接变化的 actionintentFilter = new IntentFilter();intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");// 初始化广播接收器networkChangeReceiver = new NetworkChangeReceiver();// 注册广播接收器以接收网络连接变化的广播registerReceiver(networkChangeReceiver, intentFilter);}@Overrideprotected void onDestroy() {super.onDestroy();// 在 Activity 销毁时取消注册广播接收器,避免内存泄漏unregisterReceiver(networkChangeReceiver);}// 内部类:广播接收器,用于处理网络连接变化的广播class NetworkChangeReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 获取 ConnectivityManager 实例ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);// 获取当前活动的网络信息NetworkInfo activeNetworkInfo = connectivityManager.getActiveNetworkInfo();// 检查网络是否可用if (activeNetworkInfo != null && activeNetworkInfo.isAvailable()) {// 网络可用时,显示提示Toast.makeText(context, "Network is available", Toast.LENGTH_SHORT).show();} else {// 网络不可用时,显示提示Toast.makeText(context, "Network is unavailable", Toast.LENGTH_SHORT).show();}}}
}

IntentFilter

  • IntentFilter 是 Android 中用于描述一个广播接收器感兴趣的 Intent 类型的类。IntentFilter 用于动态注册广播接收器,指明该接收器对哪些广播感兴趣。
  • 在代码中,我们创建了一个 IntentFilter 对象 intentFilter 并添加了 CONNECTIVITY_CHANGE action,这样广播接收器就会监听网络连接变化的广播。

NetworkChangeReceiver内部类:

  • NetworkChangeReceiver 继承自 BroadcastReceiver,用于接收和处理广播。这个内部类包含一个 onReceive 方法,当接收到匹配 IntentFilter 中定义的广播时会被调用。
  • onReceive 方法中,我们使用 ConnectivityManager 获取当前网络信息,并显示网络状态(可用或不可用)的提示。
  1. AndroidManifest加入下面代码,给予权限
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

静态注册

可以在程序未启动的情况下,onCreate方法运行之前接收到广播

新建一个广播接收器,并且as自动在AndroidManifest注册

 		<receiverandroid:name=".BootCompleteReceiver"android:enabled="true"android:exported="true"><intent-filter><action android:name="android.intent.action.BOOT_COMPLETED" /></intent-filter></receiver>
image-20240724115946956
public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "你开机了!!!", Toast.LENGTH_LONG).show();}
}

我们这里是一个开机启动,所以还需要权限

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

自定义广播

标准广播

image-20240724221531945

发送广播

public class MainActivity extends AppCompatActivity {private IntentFilter intentFilter;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 获取布局中的按钮并设置点击事件监听器Button button = findViewById(R.id.btn);button.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 创建一个 Intent,指定广播的动作Intent intent = new Intent("com.example.broadcastspractice2_724.MY_BROADCAST");// 设置广播只发送给当前应用的接收器intent.setPackage(getPackageName());// 发送广播sendBroadcast(intent);}});}
}

定义广播接收器

public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "在 MyBroadcastReceiver 中接收", Toast.LENGTH_LONG).show();}
}

注册广播接收器

AndroidManifest.xml 文件中的 <receiver> 元素用于声明一个广播接收器,使其能够接收特定的广播意图 (Intent)。你提供的 <receiver> 元素声明了一个名为 AnotherBroadcastReceiver 的广播接收器,并配置了它要接收的广播意图。

<receiverandroid:name=".AnotherBroadcastReceiver"android:enabled="true"android:exported="true"><intent-filter><action android:name="com.example.broadcastspractice2_724.MY_BROADCAST" /></intent-filter>
</receiver>

有序广播

image-20240724221558572

修改发送方法

sendOrderedBroadcast(intent,null);

定义第二个广播接收器

public class AnotherBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "在 AnotherBroadcastReceiver 中接收", Toast.LENGTH_SHORT).show();}
}

注册广播接收器

 		<receiverandroid:name=".AnotherBroadcastReceiver"android:enabled="true"android:exported="true"><intent-filter><action android:name="com.example.broadcastspractice2_724.MY_BROADCAST" /></intent-filter></receiver>

广播截断

  1. 设置优先级

多个接收器处理有序广播的顺序规则为

  • 优先级越大的接收器,越早收到有序广播;
  • 优先级相同的时候,越早注册的接收器越早收到有序广播
<receiverandroid:name=".MyBroadcastReceiver"android:enabled="true"android:exported="true"><intent-filter android:priority="100"><action android:name="com.example.broadcastspractice2_724.MY_BROADCAST" /></intent-filter>
</receiver>
  1. 调用方法
public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "在 MyBroadcastReceiver 中接收", Toast.LENGTH_SHORT).show();//表示将这条广播截断abortBroadcast();}
}

使用本地广播

只能使用动态注册的方式

用到LocalBroadcastManager

//使用getInstance方法获取到实例localBroadcastManager = LocalBroadcastManager.getInstance();//注册广播接收器Intent intent = new Intent();intent.setAction("com.example.broadcastspractice3");localBroadcastManager.registerReceiver(localBroadcastManager, intent);

实践-强制下线

使用ActivityCollector管理所有活动

首先需要一个ActivityCollector来管理所有的活动

public class ActivityCollector {public static List<Activity> activities=new ArrayList<>();public static void addActivity(Activity activity){activities.add(activity);}public static void removeActivity(Activity activity){activities.remove(activity);}public static void finishAll(){for (Activity activity:activities){if (!activity.isFinishing()){activity.finish();}}activities.clear();}
}

然后让所以的活动继承自定义类BaseActivityBaseActivity继承AppCompatActivity

public class BaseActivity extends AppCompatActivity {private ForcerOfflineReceiver receiver;@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}@Overrideprotected void onDestroy() {super.onDestroy();ActivityCollector.removeActivity(this);}
}

登录界面

然后编写登录界面布局文件,代码在这里省略

Screenshot_2024-07-24-20-51-03-482_com.example.fo

登录逻辑在这里省略,登录成功的逻辑:进入FirstActivity主界面

  Intent intent = new Intent(MainActivity.this, FirstActivity.class);startActivity(intent);finish();

这个finish()方法的作用是结束当前的MainActivity。也就是说,MainActivity会从活动堆栈中移除,并且用户无法通过按返回键返回到这个活动。这样可以确保当用户成功登录后,不会返回到登录界面。

主界面

这里我们就实现点击按钮后强制下线,返回到登录界面

image-20240724205507603

点击按钮后发送广播

    public void ForcedOffline(View view) {Intent intent = new Intent("com.example.forcedofflinepractice_724.FORCE_OFFLINE");sendBroadcast(intent);}

BaseActivity

我们补充BaseActivity

public class BaseActivity extends AppCompatActivity {// 定义一个广播接收器变量private ForcerOfflineReceiver receiver;@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);// 在活动创建时,将当前活动添加到活动管理器中ActivityCollector.addActivity(this);}@Overrideprotected void onDestroy() {super.onDestroy();// 在活动销毁时,从活动管理器中移除当前活动ActivityCollector.removeActivity(this);}@Overrideprotected void onResume() {super.onResume();// 创建一个 IntentFilter 用于监听指定的广播动作IntentFilter intentFilter = new IntentFilter();// 添加广播动作,用于接收 "FORCE_OFFLINE" 的广播intentFilter.addAction("com.example.forcedofflinepractice_724.FORCE_OFFLINE");// 初始化广播接收器receiver = new ForcerOfflineReceiver();// 注册广播接收器,以便接收和处理广播//第三个参数用于指定广播接收器的导出状态,这里需要设置为Context.RECEIVER_EXPORTED才能使用registerReceiver(receiver, intentFilter, Context.RECEIVER_EXPORTED);}@Overrideprotected void onPause() {super.onPause();// 在活动暂停时,取消注册广播接收器if (receiver != null) {unregisterReceiver(receiver);receiver = null;}}// 自定义广播接收器类,用于处理强制下线广播class ForcerOfflineReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 创建并配置一个警告对话框AlertDialog.Builder builder = new AlertDialog.Builder(context);builder.setTitle("Warning");builder.setMessage("你被强制下线了"); builder.setCancelable(false); builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {// 用户点击 "OK" 按钮后,执行以下操作ActivityCollector.finishAll(); // 结束所有活动// 启动 MainActivity 重新引导用户登录Intent intent1 = new Intent(context, MainActivity.class);context.startActivity(intent1);}});builder.show();}}
}

在上文的动态注册中,我们在onCreate中注册广播接收器,在onDestory中取消注册。但在这里却是在onResume注册,onPause取消注册?

处于onResume生命周期的活动即将用户可见,而我们希望接收到强制下线广播的活动是在用户正在使用的活动上,不可见的界面接收到广播没有意义。

所以我们要让每一个即将显示的活动能够接收到广播实现功能,即将不可见的活动取消广播接收器的注册。

1721821657349
  1. onResumeonPause
  • 注册:在 onResume 中注册广播接收器时,表示你希望在活动恢复到前台时开始接收广播。这通常是因为你只需要在活动处于用户可见状态时接收广播。
  • 注销:在 onPause 中注销广播接收器时,表示当活动不再处于前台时,你希望停止接收广播。这有助于避免在活动不活跃时接收广播,从而节省资源并避免潜在的内存泄漏。

适用于需要在活动可见时接收广播,而在活动不再可见时不再接收广播的情况。例如,如果你希望在用户使用应用时能够接收更新或消息,而在用户离开应用时不再接收,则这种做法是合适的。

  1. onCreateonDestroy
  • 注册:在 onCreate 中注册广播接收器时,表示你希望在活动创建时开始接收广播,不论活动是否处于前台。
  • 注销:在 onDestroy 中注销广播接收器时,表示在活动销毁时停止接收广播。这种做法通常用于应用级别的广播接收器,即使活动不处于前台时也希望接收广播。

onPause

  • 注册:在 onResume 中注册广播接收器时,表示你希望在活动恢复到前台时开始接收广播。这通常是因为你只需要在活动处于用户可见状态时接收广播。
  • 注销:在 onPause 中注销广播接收器时,表示当活动不再处于前台时,你希望停止接收广播。这有助于避免在活动不活跃时接收广播,从而节省资源并避免潜在的内存泄漏。

适用于需要在活动可见时接收广播,而在活动不再可见时不再接收广播的情况。例如,如果你希望在用户使用应用时能够接收更新或消息,而在用户离开应用时不再接收,则这种做法是合适的。

  1. onCreateonDestroy
  • 注册:在 onCreate 中注册广播接收器时,表示你希望在活动创建时开始接收广播,不论活动是否处于前台。
  • 注销:在 onDestroy 中注销广播接收器时,表示在活动销毁时停止接收广播。这种做法通常用于应用级别的广播接收器,即使活动不处于前台时也希望接收广播。

适用于需要在整个活动生命周期内持续接收广播的情况。例如,如果你希望在活动创建到销毁的整个期间都能够接收广播消息,则可以在 onCreateonDestroy 中进行注册和注销。



感谢您的阅读
如有错误烦请指正


参考:

  1. 126-广播的静态注册_哔哩哔哩_bilibili
  2. 【Android】广播BroadcastReceiver、接收系统广播(动态、静态注册方式)、发送自定义广播(发送有序广播、发送标准广播)、BroadcastReceiver实践——强制下线功能_android 接收广播-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/875117.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

sql注入 mysql 执行命令 sql注入以及解决的办法

我们以前很可能听过一个词语叫做SQL注入攻击&#xff0c;其是威胁我们系统安全的最危险的因素之一&#xff0c;那么到底什么是SQL注入攻击呢&#xff1f;这里我会用一个最经典最简单的例子来跟大家解释一下&#xff1a; 众所周知&#xff0c;我们的sql语句都是有逻辑的&#xf…

STM32之九:ADC模数转换器

目录 1. 简介 2. ADC 2.1 逐次逼近型寄存器SAR 2.2 ADC转换时间 3 ADC框图 3.1 8 bit ADC0809芯片内部框图 3.2 ADC框图 3.2.1 注入通道和规则通道 3.2.2 单次/连续转换模式 3.2.3 扫描模式 3.2.4 外部触发转换 3.2.5 数据对齐 3.2.6 模拟看门狗 4. 总结和ADC驱…

MYSQL ODBC驱动安装时的注意事项

今天想使用MYSQL的ODBC驱动连接数据库。 安装的时候遇到一个大坑&#xff0c;在这里记录一下。 window 64位的操作&#xff0c;要安装64位驱动&#xff0c;这个大家都知道了。 有以下的问题要注意区别的。 1 、windows是64位的&#xff0c;但是开发软件是32位的。 这个时候…

OpenStack Yoga版安装笔记(七)glance练习补充

1、练习场景说明 在OpenStack Yoga版安装笔记&#xff08;五&#xff09;中&#xff0c;glance已经在controller node虚拟机上安装完成&#xff0c;并且已经成功拍摄了快照。 此时&#xff0c;controller node虚机已经安装了keystone、keystone DB、glance、glance DB、OpenSta…

PCL-基于FPFH的SAC-IA结合ICP的点云配准方法

目录 一、相关方法原理1.凸包方法2.FPFH特征描述3.SAC-IA概述4.ICP概述 二、实验代码三、实验结果 一、相关方法原理 点云是在同一空间参考系下表达目标空间分布和目标表面特性的海量点集合&#xff0c;在获取物体表面每个采样点的空间坐标后&#xff0c;得到的是点的集合&…

构建智能运维系统:创新架构与效率优化

随着信息技术的迅猛发展&#xff0c;企业对于运维效率和服务质量的要求越来越高。智能运维系统的设计和实施&#xff0c;不仅能够提升系统可靠性和响应速度&#xff0c;还能有效降低成本和人力投入。本文将深入探讨智能运维系统的架构设计原则和关键技术&#xff0c;为企业在运…

数据结构重置版(概念篇)

本篇文章是对数据结构的重置&#xff0c;且只涉及概念 顺序表与链表的区别 不同点 顺序表 链表 存储空间上 物理上一定连续 逻辑上连续&#xff0c;但物理上不一定连续…

.env.local 配置本地环境变量 用于团队开发

.env.local 用途&#xff1a;.env.local 通常用于存储本地开发环境中的环境变量。这些变量可能包括敏感数据或特定于单个开发者的设置&#xff0c;不应该被提交到版本控制系统中。优先级&#xff1a;在大多数框架中&#xff0c;.env.local 文件中的变量会覆盖其他 .env 文件中…

分类模型的完整流程及Python实现

1、加载函数和数据集 import numpy as np from sklearn.datasets import load_breast_cancer from sklearn.svm import SVC from sklearn.model_selection import train_test_split from sklearn.preprocessing import StandardScaler import matplotlib.pyplot as plt cancer…

linux系统查历史cpu使用数据(使用sar 查询cpu和网络占用最近1个月历史数据)。

一 sar 指令介绍 在 Linux 系统中&#xff0c;sar 是 System Activity Reporter 的缩写&#xff0c;是一个用于收集、报告和保存系统活动信息的工具。它是 sysstat 软件包的一部分&#xff0c;提供了丰富的系统性能数据&#xff0c;包括 CPU、内存、网络、磁盘等使用情况&am…

SQL中的LEFT JOIN、RIGHT JOIN和INNER JOIN

在SQL中&#xff0c;JOIN操作是连接两个或多个数据库表&#xff0c;并根据两个表之间的共同列&#xff08;通常是主键和外键&#xff09;返回数据的重要方法。其中&#xff0c;LEFT JOIN&#xff08;左连接&#xff09;、RIGHT JOIN&#xff08;右连接&#xff09;和INNER JOIN…

《JavaEE篇》--多线程(2)

《JavaEE篇》--多线程(1) 线程安全 线程不安全 我们先来观察一个线程不安全的案例&#xff1a; public class Demo {private static int count 0;public static void main(String[] args) throws InterruptedException {Thread t1 new Thread(() -> {//让count自增5W次…

HarmonyOS网络请求的简单用法,HttpUtil简单封装

请求网络获取数据 点击按钮发送一个post请求&#xff0c;发送一条string由于此处的返回result.data本身就是一个string&#xff0c;因此不需要转换类型 Button(请求网络).margin({ top: 10 }).fontSize(24).fontWeight(FontWeight.Bold).onClick(() > {httpRequestPost(http…

风格迁移开发记录(DCT-Net)

1.DCT-Net部署 阿里旗下的 modelscope社区&#xff0c;丰富的开源风格迁移算法模型 DCT-Net GitHub链接 git clone https://github.com/menyifang/DCT-Net.git cd DCT-Netpython run_sdk.py下载不同风格的模型如下图每个文件夹代表一种风格&#xff0c;有cartoon_bg.pb, car…

C++STL详解(一)——String接口详解(上)!!!

目录 一.string类介绍 二.string类的构造赋值 2.1string类的拷贝和构造函数 2.2深拷贝 三.string类的插入 3.1push_back 3.2append 3.3操作符 3.4insert 四.string的删除 4.1pop_back 4.2erase 五.string的查找 5.1find 5.2rfind 六.string的比较 6.1compare函…

深入浅出WebRTC—Pacer

平滑发包&#xff08;Pacer&#xff09;是 WebRTC 实现高质量实时通信不可或缺的一部分。在视频通信中&#xff0c;单帧视频可能包含大量的数据&#xff0c;如果未经控制地立即发送&#xff0c;可能瞬间对网络造成巨大压力。Pacer 能够根据网络条件动态调整发送速率&#xff0c…

python库(14):Arrow库简化时间处理

1 Arrow简介 Arrow 是一个被称为程序员的时间处理利器的 Python 库。 从诞生起&#xff0c;它就是为了填补 Python 的 datetime 类型的功能空白而生的。为程序员提供了一种更简单、更直观的方式来处理日期和时间。 2 安装Arrow库 pip install arrow -i https://pypi.tuna.ts…

什么是设备运维管理系统?有什么作用?(6款设备运维管理系统推荐)

一、什么是设备运维管理系统&#xff1f; 设备运维管理系统是一种集成了监控、管理、维护和优化设备性能的软件平台。它旨在通过自动化的手段&#xff0c;提高设备运行的可靠性和效率&#xff0c;降低运维成本&#xff0c;并优化资源利用。 设备运维管理系统能够实时监控设备…

【1】Python机器学习之基础概念

1、什么是机器学习 最早的机器学习应用——垃圾邮件分辨 传统的计算机解决问题思路&#xff1a; 编写规则&#xff0c;定义“垃圾邮件”&#xff0c;让计算机执行对于很多问题&#xff0c;规则很难定义规则不断变化 机器学习在图像识别领域的重要应用&#xff1a; 人脸识别…

带您详细了解安全漏洞的产生和防护

什么是漏洞&#xff1f; 漏洞是 IT、网络、云、Web 或移动应用程序系统中的弱点或缺陷&#xff0c;可能使其容易受到成功的外部攻击。攻击者经常试图寻找网络安全中的各种类型的漏洞来组合和利用系统。 一些最常见的漏洞&#xff1a; 1.SQL注入 注入诸如 SQL 查询之类的小代…