当前位置: 首页 > news >正文

【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博客

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 云计算实训12——配置web服务器、配置客户端服务器、配置DNS服务、实现DNS域名解析
  • 深入理解算数表达式求值:后缀表达式的转换与计算
  • Python爬虫实战案例(爬取图片)
  • 【MySQL进阶之路 | 高级篇】redo日志和undo日志
  • 以太坊的可扩展性危机:探索执行层的瓶颈
  • DNS续集
  • 【时时三省】unity test 测试框架 下载
  • 【日常记录】【JS】对一个数组,按照某个字段的值,进行分组
  • Flutter 使用 url_launcher的canLaunchUrl() 方法总是返回false错误
  • 如何用JavaScript实现视频观看时间追踪
  • lua 游戏架构 之 游戏 AI (六)ai_auto_skill
  • PCB工艺边设计准则
  • WebRTC与orange pi实现视频画面实时传输
  • arinc664总线协议
  • HarmonyOS Next 省市区级联(三级联动)筛选框
  • C++回声服务器_9-epoll边缘触发模式版本服务器
  • css选择器
  • gitlab-ci配置详解(一)
  • HTML-表单
  • js数组之filter
  • PAT A1092
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • Travix是如何部署应用程序到Kubernetes上的
  • 从输入URL到页面加载发生了什么
  • 代理模式
  • 机器学习中为什么要做归一化normalization
  • 面试总结JavaScript篇
  • 使用Gradle第一次构建Java程序
  • 白色的风信子
  • 如何正确理解,内页权重高于首页?
  • #鸿蒙生态创新中心#揭幕仪式在深圳湾科技生态园举行
  • (21)起落架/可伸缩相机支架
  • (3) cmake编译多个cpp文件
  • (el-Transfer)操作(不使用 ts):Element-plus 中 Select 组件动态设置 options 值需求的解决过程
  • (LeetCode C++)盛最多水的容器
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (二)Pytorch快速搭建神经网络模型实现气温预测回归(代码+详细注解)
  • (二)丶RabbitMQ的六大核心
  • (附源码)ssm高校运动会管理系统 毕业设计 020419
  • (含笔试题)深度解析数据在内存中的存储
  • (论文阅读22/100)Learning a Deep Compact Image Representation for Visual Tracking
  • (免费领源码)python+django+mysql线上兼职平台系统83320-计算机毕业设计项目选题推荐
  • (一)pytest自动化测试框架之生成测试报告(mac系统)
  • (译) 函数式 JS #1:简介
  • (转)Android学习系列(31)--App自动化之使用Ant编译项目多渠道打包
  • (转)memcache、redis缓存
  • (转)Unity3DUnity3D在android下调试
  • (转)一些感悟
  • .[hudsonL@cock.li].mkp勒索加密数据库完美恢复---惜分飞
  • .“空心村”成因分析及解决对策122344
  • .MSSQLSERVER 导入导出 命令集--堪称经典,值得借鉴!
  • .NET Framework Client Profile - a Subset of the .NET Framework Redistribution
  • .NET 中使用 TaskCompletionSource 作为线程同步互斥或异步操作的事件
  • .NET版Word处理控件Aspose.words功能演示:在ASP.NET MVC中创建MS Word编辑器
  • .NET开发不可不知、不可不用的辅助类(一)