深入分析 Android BroadcastReceiver (一)

2024-06-02 09:20

本文主要是介绍深入分析 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() {@Overridepublic void onReceive(Context context, Intent intent) {// 处理网络变化的逻辑}};@Overrideprotected void onStart() {super.onStart();IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);registerReceiver(networkReceiver, filter);}@Overrideprotected void onStop() {super.onStop();unregisterReceiver(networkReceiver);}
}

3. BroadcastReceiver 的生命周期

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

public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic 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 {@Overridepublic 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() {@Overridepublic 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 {@Overridepublic 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 {@Overridepublic void onReceive(Context context, Intent intent) {// 处理本地广播}
}// 动态注册和取消注册
public class MainActivity extends AppCompatActivity {private LocalReceiver localReceiver;private LocalBroadcastManager localBroadcastManager;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);localBroadcastManager = LocalBroadcastManager.getInstance(this);localReceiver = new LocalReceiver();}@Overrideprotected void onStart() {super.onStart();IntentFilter filter = new IntentFilter("com.example.LOCAL_ACTION");localBroadcastManager.registerReceiver(localReceiver, filter);}@Overrideprotected 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 {@Overridepublic 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 的使用方法,并将其应用于实际项目中。

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

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

在这里插入图片描述

这篇关于深入分析 Android BroadcastReceiver (一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1023566

相关文章

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

android应用中res目录说明

Android应用的res目录是一个特殊的项目,该项目里存放了Android应用所用的全部资源,包括图片、字符串、颜色、尺寸、样式等,类似于web开发中的public目录,js、css、image、style。。。。 Android按照约定,将不同的资源放在不同的文件夹中,这样可以方便的让AAPT(即Android Asset Packaging Tool , 在SDK的build-tools目

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使

Android Environment 获取的路径问题

1. 以获取 /System 路径为例 /*** Return root of the "system" partition holding the core Android OS.* Always present and mounted read-only.*/public static @NonNull File getRootDirectory() {return DIR_ANDR

Android逆向(反调,脱壳,过ssl证书脚本)

文章目录 总结 基础Android基础工具 定位关键代码页面activity定位数据包参数定位堆栈追踪 编写反调脱壳好用的脚本过ssl证书校验抓包反调的脚本打印堆栈bilibili反调的脚本 总结 暑假做了两个月的Android逆向,记录一下自己学到的东西。对于app渗透有了一些思路。 这两个月主要做的是代码分析,对于分析完后的持久化等没有学习。主要是如何反编译源码,如何找到

android系统源码12 修改默认桌面壁纸--SRO方式

1、aosp12修改默认桌面壁纸 代码路径 :frameworks\base\core\res\res\drawable-nodpi 替换成自己的图片即可,不过需要覆盖所有目录下的图片。 由于是静态修改,则需要make一下,重新编译。 2、方法二Overlay方式 由于上述方法有很大缺点,修改多了之后容易遗忘自己修改哪些文件,为此我们采用另外一种方法,使用Overlay方式。