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

server/2024/10/22 15:28:10/
xmlns="http://www.w3.org/2000/svg" style="display: none;">

文章目录

  • 动态注册
    • 使用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加入下面代码,给予权限
xml">    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

静态注册

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

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

xml"> 		<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();}
}

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

xml">    <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 的广播接收器,并配置了它要接收的广播意图。

xml"><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();}
}

注册广播接收器

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

广播截断

  1. 设置优先级

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

  • 优先级越大的接收器,越早收到有序广播;
  • 优先级相同的时候,越早注册的接收器越早收到有序广播
xml"><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.ppmy.cn/server/90114.html

相关文章

LLM 大语言模型显存消耗估计与计算

LLM 大语言模型显存消耗估计与计算 1. LLM 大语言模型开发流程 在大模型&#xff08;如 LLaMA-7B、GPT-3 等&#xff09;的开发、训练、微调、推理和部署过程中&#xff0c;各个阶段的流程都涉及多个复杂的步骤。以下是详细的流程描述&#xff0c;涵盖训练和微调的区别&#…

MySQL数据库(基础篇)

&#x1f30f;个人博客主页&#xff1a;心.c 前言&#xff1a;今天讲解的是MySQL的详细知识点的&#xff0c;希望大家可以收货满满&#xff0c;话不多说&#xff0c;直接开始搞&#xff01; &#x1f525;&#x1f525;&#x1f525;文章专题&#xff1a;MySQL &#x1f63d;感…

从零开始的Python开发日记(4):如何连接SQLite数据库

如何使用Python连接SQLite数据库 SQLite 是一种轻量级的嵌入式数据库&#xff0c;广泛用于小型应用和开发原型。Python 提供了内置的 sqlite3 库&#xff0c;可以方便地连接和操作 SQLite 数据库。本文将介绍如何使用 Python 连接 SQLite 数据库&#xff0c;并包含代码示例。 …

Spring 简化jdbc开发 05

文章目录 1. JdbcTemplate&#xff1a;2.命名参数和占位符3. 异常处理&#xff1a;4. 事务管理&#xff1a; 在 Spring 框架中&#xff0c;通过其提供的 JDBC 抽象和工具类&#xff0c;可以极大地简化 JDBC 的开发过程&#xff0c;减少了传统 JDBC 开发中的样板代码和错误处理。…

微信答题小程序产品研发-UI界面设计

高保真原型虽然已经很接近产品形态了&#xff0c;但毕竟还不能够直接交付给开发。这时就需要UI设计师依据之前的原型设计&#xff0c;进一步细化和实现界面的视觉元素&#xff0c;包括整体视觉风格、颜色、字体、图标、按钮以及交互细节优化等。 UI设计不仅关系到用户的直观感…

OAK-FFC 分体式相机使用入门介绍

概述 OAK FFC 主控板和多种可选配镜头模组非常适合灵活的搭建您的3D人工智能产品原型。由于镜头是分体式的&#xff0c;因此你可以根据需要测量的距离&#xff0c;自定义深度相机安装基线&#xff0c;并根据你的项目要求&#xff08;分辨率、快门类型、FPS、光学元件&#xff…

HarmonyOS(45) 控件拖动或者拖拽PanGesture

PanGesture实现控件拖动的效果&#xff0c;通过拖动的坐标位置调用position或者translate方法来更新UI的位置。效果见下图&#xff1a; 具体代码如下&#xff1a; // xxx.ets Entry Component struct PanGestureExample {State offsetX: number 0State offsetY: number 0pos…

多线程死锁与单例模式

目录 1.两个线程两把锁的死锁场景 2.构成死锁的四个必要条件&#xff1a; 3.内存可见性问题 4.volatile的作用 5.wait与sleep的区别 6.单例模式 1.两个线程两把锁的死锁场景 1&#xff09;线程1先针对线程A加锁&#xff0c;线程2针对线程B加锁&#xff0c;线程1在不释放…