深入分析 Android BroadcastReceiver (一)

文章目录

    • 深入分析 Android BroadcastReceiver (一)
    • 1. Android BroadcastReceiver 设计说明
      • 1.1 BroadcastReceiver 的主要用途
    • 2. BroadcastReceiver 的工作机制
      • 2.1 注册 BroadcastReceiver
        • 2.1.1 静态注册
        • 2.1.2 动态注册
    • 3. BroadcastReceiver 的生命周期
    • 4. 实现和使用 BroadcastReceiver 的步骤
    • 5. 安全性与权限管理
    • 6. BroadcastReceiver 的高级使用与优化
      • 6.1 高级使用场景
      • 6.2 性能优化
      • 6.3 实战示例
    • 7. 总结

深入分析 Android BroadcastReceiver (一)

1. Android BroadcastReceiver 设计说明

BroadcastReceiver 是 Android 四大组件之一,用于接收并处理广播消息。在 Android 中,广播是一种跨进程通信的机制,允许应用程序发送和接收系统或应用级别的广播消息。

1.1 BroadcastReceiver 的主要用途

  • 系统广播:接收系统发出的广播,例如网络状态变化、电池电量低等。
  • 应用内部广播:应用内部组件之间的通信,例如通知其他组件数据下载完成。
  • 跨应用广播:在不同应用之间发送广播,进行通信。

2. BroadcastReceiver 的工作机制

BroadcastReceiver 通过注册来接收特定的广播事件,当系统或应用程序发送相应的广播时,BroadcastReceiveronReceive 方法被调用。

2.1 注册 BroadcastReceiver

广播接收器可以通过两种方式注册:静态注册和动态注册。

2.1.1 静态注册

静态注册是在 AndroidManifest.xml 文件中声明的,适用于应用未启动时也需要接收的广播。例如,监听设备启动完成的广播:

<receiver android:name=".BootCompletedReceiver">
    <intent-filter>
        <action android:name="android.intent.action.BOOT_COMPLETED" />
    </intent-filter>
</receiver>
2.1.2 动态注册

动态注册是在代码中进行的,通常在 Activity 或 Service 的生命周期方法中进行注册,并在适当的时候取消注册。例如,监听网络变化的广播:

public class MainActivity extends AppCompatActivity {
    private BroadcastReceiver networkReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 处理网络变化的逻辑
        }
    };

    @Override
    protected void onStart() {
        super.onStart();
        IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);
        registerReceiver(networkReceiver, filter);
    }

    @Override
    protected void onStop() {
        super.onStop();
        unregisterReceiver(networkReceiver);
    }
}

3. BroadcastReceiver 的生命周期

BroadcastReceiver 的生命周期非常短暂,仅在 onReceive 方法执行期间有效。广播接收器不能执行耗时操作,如网络请求或数据库操作。如果需要执行这些操作,应该启动一个 Service 或使用 JobScheduler 进行处理。

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // 启动服务来处理耗时操作
        Intent serviceIntent = new Intent(context, MyIntentService.class);
        context.startService(serviceIntent);
    }
}

4. 实现和使用 BroadcastReceiver 的步骤

  1. 创建 BroadcastReceiver

    • 创建一个继承自 BroadcastReceiver 的类,并重写 onReceive 方法。
    public class MyBroadcastReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
            if (action != null) {
                switch (action) {
                    case ConnectivityManager.CONNECTIVITY_ACTION:
                        // 处理网络变化
                        break;
                    case Intent.ACTION_BATTERY_LOW:
                        // 处理电池低电量
                        break;
                }
            }
        }
    }
    
  2. 注册 BroadcastReceiver

    • 在 AndroidManifest.xml 文件中进行静态注册,或在代码中进行动态注册。
    <receiver android:name=".MyBroadcastReceiver">
        <intent-filter>
            <action android:name="android.intent.action.BATTERY_LOW" />
            <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
        </intent-filter>
    </receiver>
    
    @Override
    protected void onStart() {
        super.onStart();
        IntentFilter filter = new IntentFilter();
        filter.addAction(Intent.ACTION_BATTERY_LOW);
        filter.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
        registerReceiver(myBroadcastReceiver, filter);
    }
    
    @Override
    protected void onStop() {
        super.onStop();
        unregisterReceiver(myBroadcastReceiver);
    }
    
  3. 发送广播

    • 通过 Context.sendBroadcast 方法发送广播。
    Intent intent = new Intent("com.example.CUSTOM_ACTION");
    sendBroadcast(intent);
    

5. 安全性与权限管理

为了提高安全性,广播接收器可以声明权限,以确保只有具有相应权限的应用程序才能发送或接收广播。

<receiver android:name=".MyBroadcastReceiver">
    <intent-filter>
        <action android:name="com.example.CUSTOM_ACTION" />
    </intent-filter>
    <permission android:name="com.example.MY_PERMISSION" />
</receiver>
Intent intent = new Intent("com.example.CUSTOM_ACTION");
sendBroadcast(intent, "com.example.MY_PERMISSION");

6. BroadcastReceiver 的高级使用与优化

在深入理解 BroadcastReceiver 的基础上,可以进一步探讨其高级使用场景与优化策略,以便在实际项目中更好地应用该组件。

6.1 高级使用场景

  1. 本地广播(LocalBroadcastManager)

    • 使用本地广播可以避免跨进程通信的开销,同时增加安全性,防止其他应用截获广播消息。
    // 发送本地广播
    LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(context);
    Intent localIntent = new Intent("com.example.LOCAL_ACTION");
    localBroadcastManager.sendBroadcast(localIntent);
    
    // 接收本地广播
    LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(context);
    BroadcastReceiver localReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 处理本地广播
        }
    };
    IntentFilter localFilter = new IntentFilter("com.example.LOCAL_ACTION");
    localBroadcastManager.registerReceiver(localReceiver, localFilter);
    
  2. 有序广播(Ordered Broadcasts)

    • 有序广播允许多个接收器按照优先级顺序接收广播,并且可以中断广播的传播。
    // 发送有序广播
    Intent intent = new Intent("com.example.ORDERED_ACTION");
    sendOrderedBroadcast(intent, null);
    
    // 接收有序广播
    public class MyOrderedBroadcastReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 处理有序广播
        }
    }
    
    <receiver android:name=".MyOrderedBroadcastReceiver" android:priority="100">
        <intent-filter>
            <action android:name="com.example.ORDERED_ACTION" />
        </intent-filter>
    </receiver>
    
  3. 粘性广播(Sticky Broadcasts)

    • 粘性广播已被弃用,但仍然可以在某些特定场景下使用。它允许广播在发送后被系统保留,后续接收器可以获取到最后一次的广播内容。
    // 发送粘性广播
    Intent intent = new Intent("com.example.STICKY_ACTION");
    sendStickyBroadcast(intent);
    
    // 接收粘性广播
    IntentFilter filter = new IntentFilter("com.example.STICKY_ACTION");
    registerReceiver(stickyReceiver, filter);
    
    // 清除粘性广播
    removeStickyBroadcast(intent);
    

6.2 性能优化

  1. 避免耗时操作

    • onReceive 方法中执行耗时操作会阻塞广播的处理,导致应用卡顿。可以通过启动 Service 或使用 AsyncTask 来处理耗时操作。
    @Override
    public void onReceive(Context context, Intent intent) {
        Intent serviceIntent = new Intent(context, MyIntentService.class);
        context.startService(serviceIntent);
    }
    
  2. 注册和取消注册

    • 动态注册的 BroadcastReceiver 应在合适的生命周期方法中注册和取消注册,以避免内存泄漏和无用的广播接收。
    @Override
    protected void onStart() {
        super.onStart();
        registerReceiver(myReceiver, new IntentFilter("com.example.MY_ACTION"));
    }
    
    @Override
    protected void onStop() {
        super.onStop();
        unregisterReceiver(myReceiver);
    }
    
  3. 使用本地广播

    • 尽量使用 LocalBroadcastManager 发送和接收广播,减少跨进程通信的开销。
  4. 权限管理

    • 对于敏感的广播,使用权限管理来限制发送和接收广播的应用,确保安全性。
    Intent intent = new Intent("com.example.SENSITIVE_ACTION");
    sendBroadcast(intent, "com.example.SENSITIVE_PERMISSION");
    

6.3 实战示例

以下是一个综合示例,展示了如何使用本地广播、有序广播以及优化策略:

// 本地广播接收器
public class LocalReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // 处理本地广播
    }
}

// 动态注册和取消注册
public class MainActivity extends AppCompatActivity {
    private LocalReceiver localReceiver;
    private LocalBroadcastManager localBroadcastManager;

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

        localBroadcastManager = LocalBroadcastManager.getInstance(this);
        localReceiver = new LocalReceiver();
    }

    @Override
    protected void onStart() {
        super.onStart();
        IntentFilter filter = new IntentFilter("com.example.LOCAL_ACTION");
        localBroadcastManager.registerReceiver(localReceiver, filter);
    }

    @Override
    protected void onStop() {
        super.onStop();
        localBroadcastManager.unregisterReceiver(localReceiver);
    }

    private void sendLocalBroadcast() {
        Intent intent = new Intent("com.example.LOCAL_ACTION");
        localBroadcastManager.sendBroadcast(intent);
    }
}

// 有序广播接收器
public class OrderedReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // 处理有序广播
    }
}

// AndroidManifest.xml 中声明
<receiver android:name=".OrderedReceiver" android:priority="100">
    <intent-filter>
        <action android:name="com.example.ORDERED_ACTION" />
    </intent-filter>
</receiver>

7. 总结

BroadcastReceiver 是 Android 应用程序中处理广播消息的关键组件。理解其设计和工作机制,正确地进行静态和动态注册,以及合理地管理生命周期和安全性,是开发高效、稳定应用程序的基础。通过上述步骤和示例,开发者可以掌握 BroadcastReceiver 的使用方法,并将其应用于实际项目中。

通过合理使用本地广播、有序广播等高级特性,以及优化注册和处理过程,可以提高应用的性能和安全性。在实际项目中,开发者需要根据具体需求选择合适的广播机制,并遵循最佳实践以确保应用的稳定性和高效性。

欢迎点赞|关注|收藏|评论,您的肯定是我创作的动力

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Android下HWC以及drm_hwcomposer普法(上)

Android下HWC以及drm_hwcomposer普法(上) 引言 按摩得全套&#xff0c;错了&#xff0c;做事情得全套&#xff0c;普法分析也是如此。drm_hwcomposer如果对Android图形栈有一定研究的童鞋们应该知道它是Android提供的一个的图形后端合成处理HAL模块的实现。但是在分析这个之前…

yolov8使用:数据格式转换(目标检测、图像分类)多目标跟踪

安装 yolov8地址&#xff1a;https://github.com/ultralytics/ultralytics git clone https://github.com/ultralytics/ultralytics.git安装环境&#xff1a; pip install ultralytics -i https://pypi.tuna.tsinghua.edu.cn/simple目标检测 标注格式转换 若使用 labelimg…

sql注入-布尔盲注

布尔盲注&#xff08;Boolean Blind SQL Injection&#xff09;是一种SQL注入攻击技术&#xff0c;用于在无法直接获得查询结果的情况下推断数据库信息&#xff1b;它通过发送不同的SQL查询来观察应用程序的响应&#xff0c;进而判断查询的真假&#xff0c;并逐步推断出有用的信…

微服务学习Day9

文章目录 分布式事务seata引入理论基础CAP定理BASE理论 初识Seata动手实践XA模式AT模式TCC模式SAGA模式 高可用 分布式事务seata 引入 理论基础 CAP定理 BASE理论 初识Seata 动手实践 XA模式 AT模式 TCC模式 Service Slf4j public class AccountTCCServiceImpl implements A…

C语言 | Leetcode C语言题解之第126题单词接龙II

题目&#xff1a; 题解&#xff1a; char** list; int** back; int* backSize;// DFS uses backtrack information to construct results void dfs(char*** res, int* rSize, int** rCSizes, int* ans, int last, int retlevel) {int i ans[last];if (i 0) {res[*rSize] (c…

DALL·E 2详解:人工智能如何将您的想象力变为现实!

引言 DALLE 2是一个基于人工智能的图像生成模型&#xff0c;它通过理解自然语言描述来生成匹配这些描述的图像。这一模型的核心在于其创新的两阶段工作流程&#xff0c;首先是将文本描述转换为图像表示&#xff0c;然后是基于这个表示生成具体的图像。 下面详细介绍DALL-E2的功…

Vivado Design Suite一级物件

Vivado Design Suite一级物件 按设计过程导航内容 Xilinx文档围绕一组标准设计流程进行组织&#xff0c;以帮助您 查找当前开发任务的相关内容。本文件涵盖 以下设计过程&#xff1a; •硬件、IP和平台开发&#xff1a;为硬件创建PL IP块 平台&#xff0c;创建PL内核&#xff0…

HTML的标签(标题、段落、文本、图片、列表)

HTML的标签1 标题标签&#xff1a;段落标签&#xff1a;文本标签&#xff1a;图片标签:列表标签&#xff1a;有序列表&#xff1a;无序列表&#xff1a;定义列表&#xff1a;列表案例&#xff1a; 标题标签&#xff1a; 标签&#xff1a;h1~h6 注意&#xff1a;如果使用无效标…

C语言怎样写数据⽂件,使之可以在不同字⼤⼩、 字节顺序或浮点格式的机器上读⼊?

一、问题 怎样写数据⽂件&#xff0c;使之可以在不同字⼤⼩、字节顺序或浮点格式的机器上读⼊&#xff0c;也就是说怎样写⼀个可移植性好的数据⽂件&#xff1f; 二、解答 最好的移植⽅法是使⽤⽂本⽂件&#xff0c;它的每⼀字节放⼀个 ASCII 代码&#xff0c;代表⼀个字符。 …

从JS角度直观理解递归的本质

让我们写一个函数 pow(x, n)&#xff0c;它可以计算 x 的 n 次方。换句话说就是&#xff0c;x 乘以自身 n 次。 有两种实现方式。 迭代思路&#xff1a;使用 for 循环&#xff1a; function pow(x, n) {let result 1;// 在循环中&#xff0c;用 x 乘以 result n 次for (let i…

短时间内如何顺利通过 Java 面试?

今天我们来探讨一个重要的话题&#xff1a;短时间内如何顺利通过 Java 面试&#xff1f; 在此之前&#xff0c;我正在精心编写一套完全面向小白的 Java 自学教程&#xff0c;我相信这套教程会非常适合正在努力提升的你。教程里面涵盖了丰富全面的编程教学内容、详细生动的视频…

2.8Flowmap的实现

一、Flowmap 是什么 半条命2中水的流动 求生之路2中的水的流动 这种方式原理简单&#xff0c;容易实现&#xff0c;运算量少&#xff0c;如今也还在使用 1.flowmap的实质 Flow map(流向图) &#xff0c;一张记录了2D向量信息的纹理&#xff0c;Flow map上的颜色(通常为RG通道…

Python知识点14---被规定的资源

提前说一点&#xff1a;如果你是专注于Python开发&#xff0c;那么本系列知识点只是带你入个门再详细的开发点就要去看其他资料了&#xff0c;而如果你和作者一样只是操作其他技术的Python API那就足够了。 在Python中被规定的东西不止有常识中的那些关键字、构造器等编程语言…

Vue3-Ref Reactive toRef toRefs对比学习、标签ref与组件ref

响应式数据&#xff1a; Ref 作用&#xff1a;定义响应式变量。 语法&#xff1a;let xxx ref(初始值)(里面可以是任何规定内类型、数组等)。 返回值&#xff1a;一个RefImpl的实例对象&#xff0c;简称ref对象或ref&#xff0c;ref对象的value属性是响应式的。 注意点&am…

公网如何访问内网?

公网和内网已经成为我们生活中不可或缺的存在。由于内网的安全性考虑&#xff0c;公网无法直接访问内网资源。如何实现公网访问内网呢&#xff1f;本文将介绍一种名为【天联】的私有通道技术&#xff0c;通过安全加密&#xff0c;保障数据传输的安全性。 【天联】私有通道技术 …

利用Python处理DAX多条件替换

小A&#xff1a;白茶&#xff0c;救命啊~~~ 白茶&#xff1a;什么情况&#xff1f; 小A&#xff1a;是这样的&#xff0c;最近不是临近项目上线嘛&#xff0c;有一大波度量值需要进行类似的调整&#xff0c;一个两个倒没啥&#xff0c;600多个&#xff0c;兄弟&#xff0c;救命…

STM32_FSMC_HAL(介绍)

FSMC&#xff08;Flexible Static Memory Controller&#xff09;是STM32微控制器中的一种内存控制器&#xff0c;它允许微控制器与外部存储器接口&#xff0c;如SRAM、NOR Flash、NAND Flash和PSRAM等。FSMC特别适用于需要高速数据交换和大量数据存储的应用场景。 典型应用&a…

06.持久化存储

6.持久化存储 pv: persistent volume 全局的资源 pv&#xff0c;node pvc: persistent volume claim 局部的资源&#xff08;namespace&#xff09;pod&#xff0c;rc&#xff0c;svc 6.1:安装nfs服务端(192.168.111.11) yum install nfs-utils.x86_64 -y mkdir /data vim /…

Linux——多线程(二)

在上一篇博客中我们已经介绍到了线程控制以及对应的函数调用接口&#xff0c;接下来要讲的是真正的多线程&#xff0c;线程安全、线程互斥、同步以及锁。 一、多线程 简单写个多线程的创建、等待的代码 #include<iostream> #include<pthread.h> #include<un…

【案例实操】银河麒麟桌面操作系统实例分享,V10SP1重启后网卡错乱解决方法

1.问题现象 8 个网口&#xff0c; 命名从 eth1 开始到 eth8。 目前在系统 grub 里面加了 net.ifnames0 biosdevname0 参数&#xff0c; 然后在 udev 规则中加了一条固定网卡和硬件 pci 设备号的规则文件。 最后在 rc.local 中加了两条重新安装网卡驱动的命令&#xff08; rmmod…