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

news2024/9/22 23:36:50

文章目录

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

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

动态注册

使用BroadcastReceiver监听Intent广播

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

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

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

public class MainActivity extends AppCompatActivity {

    //...
    
    // 内部类:广播接收器,用于处理网络连接变化的广播
    class NetworkChangeReceiver extends BroadcastReceiver {
        @Override
        public 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 和 BroadcastReceiver
    private IntentFilter intentFilter;
    private NetworkChangeReceiver networkChangeReceiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        // 初始化 IntentFilter 并添加网络连接变化的 action
        intentFilter = new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        // 初始化广播接收器
        networkChangeReceiver = new NetworkChangeReceiver();
        // 注册广播接收器以接收网络连接变化的广播
        registerReceiver(networkChangeReceiver, intentFilter);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 在 Activity 销毁时取消注册广播接收器,避免内存泄漏
        unregisterReceiver(networkChangeReceiver);
    }

    // 内部类:广播接收器,用于处理网络连接变化的广播
    class NetworkChangeReceiver extends BroadcastReceiver {
        @Override
        public 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注册

 		<receiver
            android: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 {
    @Override
    public 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;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 获取布局中的按钮并设置点击事件监听器
        Button button = findViewById(R.id.btn);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public 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 {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "在 MyBroadcastReceiver 中接收", Toast.LENGTH_LONG).show();
    }
}

注册广播接收器

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

<receiver
    android: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 {

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

注册广播接收器

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

广播截断

  1. 设置优先级

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

  • 优先级越大的接收器,越早收到有序广播;
  • 优先级相同的时候,越早注册的接收器越早收到有序广播
<receiver
    android: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 {
    @Override
    public 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;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ActivityCollector.addActivity(this);
    }
    
    @Override
    protected 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;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 在活动创建时,将当前活动添加到活动管理器中
        ActivityCollector.addActivity(this);
    }
    
        @Override
    protected void onDestroy() {
        super.onDestroy();
        // 在活动销毁时,从活动管理器中移除当前活动
        ActivityCollector.removeActivity(this);
    }

    @Override
    protected 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);
        
    }

    @Override
    protected void onPause() {
        super.onPause();
        // 在活动暂停时,取消注册广播接收器
        if (receiver != null) {
            unregisterReceiver(receiver);
            receiver = null;
        }
    }

    // 自定义广播接收器类,用于处理强制下线广播
    class ForcerOfflineReceiver extends BroadcastReceiver {
        @Override
        public 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() {
                @Override
                public 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.coloradmin.cn/o/1946761.html

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

相关文章

ubuntu那些ppa源在哪

Ubuntu中的 PPA 终极指南 - UBUNTU粉丝之家 什么是PPA PPA 代表个人包存档。 PPA 允许应用程序开发人员和 Linux 用户创建自己的存储库来分发软件。 使用 PPA&#xff0c;您可以轻松获取较新的软件版本或官方 Ubuntu 存储库无法提供的软件。 为什么使用PPA&#xff1f; 正如…

【JavaEE】Spring Boot 自动装配原理(源码分析)

一. 前言 我们在写Spring Boot的程序代码的时候, 可以注入很多我们没有定义过的Bean.例如: Autowired private ApplicationContext applicationContext; Autowired public DataSourceTransactionManager transactionManager; Autowired public AutowireCapableBeanFactory …

软件开发者消除edge浏览器下载时“此应用不安全”的拦截方法

当Microsoft Edge浏览器显示“此应用不安全”或者“已阻止此不安全的下载”这类警告时&#xff0c;通常是因为Windows Defender SmartScreen或者其他安全功能认为下载的文件可能存在安全风险。对于软件开发者来说&#xff0c;大概率是由于软件没有进行数字签名&#xff0c;导致…

Visual Studio 2022新建 cmake 工程测试 tensorRT 自带样例 sampleOnnxMNIST

1. 新建 cmake 工程 vs2022_cmake_sampleOnnxMNIST_test( 如何新建 cmake 工程&#xff0c;请参考博客&#xff1a;Visual Studio 2022新建 cmake 工程测试 opencv helloworld ) 2. 删除默认生成的 vs2022_cmake_sampleOnnxMNIST_test.h 头文件 3. 修改默认生成的 vs2022_cma…

【屏显MCU】多媒体接口总结

本文主要介绍【屏显MCU】的基本概念&#xff0c;用于开发过程中的理解 以下是图层叠加示例 【屏显MCU】多媒体接口总结 0. 个人简介 && 授权须知1. 三大引擎1.1 【显示引擎】Display Engine1.1.1 【UI】 图层的概念1.1.2 【Video】 图层的概念1.1.3 图层的 Blending 的…

一键解锁:科研服务器性能匹配秘籍,选择性能精准匹配科研任务和计算需求的服务器

一键解锁&#xff1a;科研服务器性能匹配秘籍 HPC科研工作站服务器集群细分领域迷途小书童 专注于HPC科研服务器细分领域kyfwq001 &#x1f3af;在当今科技飞速发展的时代&#xff0c;科研工作对计算资源的需求日益增长&#x1f61c;。选择性能精准匹配科研任务和计算需求的服…

古籍双层PDF制作教程:保姆级古籍数字化教程

在智慧古籍数字化项目中&#xff0c;很多图书馆要求将古籍导出为双层PDF&#xff0c;并且确保输出双层PDF底层文本与上层图片偏移量控制在1毫米以内。那么本教程带你使用古籍数字化平台&#xff0c;3分钟把一个古籍书籍转化为双侧PDF。 第1步&#xff1a;上传古籍 点批量上传…

前序+中序、中序+后序构造二叉树

https://leetcode.cn/problems/construct-binary-tree-from-preorder-and-inorder-traversal/ https://leetcode.cn/problems/construct-binary-tree-from-inorder-and-postorder-traversal/ 前序中序 前序遍历&#xff0c;节点按照 [根左右] 排序。 中序遍历&#xff0c;节点…

JavaEE - Spring Boot 简介

1.Maven 1.1 什么是Maven 翻译过来就是: Maven是⼀个项⽬管理⼯具。基于POM(Project Object Model,项⽬对象模型)的概念&#xff0c;Maven可以通 过⼀⼩段描述信息来管理项⽬的构建&#xff0c;报告和⽂档的项⽬管理⼯具软件。 可以理解为&#xff1a;Maven是一个项目管理工具…

nginx隐藏server及版本号

1、背景 为了提高nginx服务器的安全性&#xff0c;降低被攻击的风险&#xff0c;需要隐藏nginx的server和版本号。 2、隐藏nginx版本号 在 http {—}里加上 server_tokens off; 如&#xff1a; http {……省略sendfile on;tcp_nopush on;keepalive_timeout 60;tcp_nodelay o…

ROS参数服务器增删改查实操Python

ROS参数服务器增删改查实操Python 环境准备参数服务器新增&#xff08;修改&#xff09;参数参数服务器获取参数参数服务器删除参数 ROS通信机制包括话题通信、服务通信和参数服务器三种通信方式&#xff0c;各原理及代码实现如下表 功能博客链接说明VScode配置 ROS 环境VScode…

《后端程序猿 · @Value 注释说明》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…

基于 HTML+ECharts 实现监控平台数据可视化大屏(含源码)

构建监控平台数据可视化大屏&#xff1a;基于 HTML 和 ECharts 的实现 监控平台的数据可视化对于实时掌握系统状态、快速响应问题至关重要。通过直观的数据展示&#xff0c;运维团队可以迅速发现异常&#xff0c;优化资源配置。本文将详细介绍如何利用 HTML 和 ECharts 实现一个…

Unity3D之TCP网络通信(客户端)

文章目录 概述TCP核心类异步机制 Unity中创建TCP客户端Unity中其它脚本获取TCP客户端接受到的数据后续改进 本文将以Unity3D应用项目作为客户端去连接制定的服务器为例进行相关说明。 Unity官网参考资料&#xff1a; https://developer.unity.cn/projects/6572ea1bedbc2a001ef…

go语言day17 通道channel

Golang-100-Days/Day16-20(Go语言基础进阶)/day18_channel通道.md at master rubyhan1314/Golang-100-Days (github.com) go语言day09 通道 协程的死锁-CSDN博客 channel for range 循环通道对象 单向通道 单项通道常用于函数参数&#xff0c;只是用来限定在函数中只能进行通道…

【Android】广播机制

【Android】广播机制 前言 广播机制是Android中一种非常重要的通信机制&#xff0c;用于在应用程序之间或应用程序的不同组件之间传递信息。广播可以是系统广播&#xff0c;也可以是自定义广播。广播机制主要包括标准广播和有序广播两种类型。 简介 在Android中&#xff0c…

【学一点儿前端】getaddrinfo ENOTFOUND registry.nlark.com“.

问题 今天jenkins打包一个项目&#xff0c;发现报错了 error An unexpected error occurred: "https://registry.nlark.com/xxxxxxxxxx.tgz: getaddrinfo ENOTFOUND registry.nlark.com". 先写解决方案 把yarn.lock文件里面的registry.nlark.com替换为registry.npmmi…

超低功耗ARM Cortex-M33 TZ MCU STM32WBA54、STM32WBA55:通过提升无线性能实现更出色的用户体验

摘要 STM32WBA54、STM32WBA55产品系列同时支持多种无线标准&#xff0c;包括Bluetooth低功耗 5.4&#xff08;已认证&#xff09;、Zigbee、Thread以及可用作Thread边界路由器的Matter。 该产品系列具有出色的灵活性和更强的安全性&#xff0c;可帮助开发人员应对不断变化的无…

二、QGroundControl开发环境搭建

文章目录 环境列表QGC源码下载编译 环境列表 QGC GithubPX4-AutopilotQt 5.15Ubuntu20.04 QGC源码下载编译 官网下载指令 如下 // Clone the repo (or your fork) including submodules: git clone --recursive -j8 https://github.com/mavlink/qgroundcontrol.git // Upda…

【vluhub】elasticsearch漏洞

Elasticsearch介绍 是Apache旗下的一个开源的、分布式、RESTful的搜索和分析引擎&#xff0c;适用于java语言项目 默认端口9200 kali中搭建ElasticHD, 即可未授权绕过ES可视化界面 直通车 https://github.com/360EntSecGroup-Skylar/ElasticHD/releases/download/1.4/elas…