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

Android Broadcast广播封装

在 Android 高版本(尤其是 Android 8.0 及以上)中,Broadcast 的使用受到了一些限制,例如 隐式广播 的限制和 后台执行限制。为了适配高版本并简化 Broadcast 的使用,可以封装一个工具类,支持动态注册、静态注册、权限控制等功能。

以下是 Broadcast 工具类的封装,支持高版本适配,并提供简洁的 API。

  1. Broadcast 工具类封装
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Build;
import androidx.annotation.RequiresApi;
import java.util.HashMap;
import java.util.Map;

public class BroadcastUtils {

    private static final Map<Context, Map<BroadcastReceiver, IntentFilter>> receiverMap = new HashMap<>();

    /**
     * 动态注册广播
     *
     * @param context  上下文
     * @param receiver 广播接收器
     * @param filter   意图过滤器
     */
    public static void registerReceiver(Context context, BroadcastReceiver receiver, IntentFilter filter) {
        if (context == null || receiver == null || filter == null) {
            return;
        }

        // 高版本适配:使用 ContextCompat 注册
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
            context.registerReceiver(receiver, filter, Context.RECEIVER_NOT_EXPORTED);
        } else {
            context.registerReceiver(receiver, filter);
        }

        // 保存注册信息,用于取消注册
        Map<BroadcastReceiver, IntentFilter> contextReceivers = receiverMap.get(context);
        if (contextReceivers == null) {
            contextReceivers = new HashMap<>();
            receiverMap.put(context, contextReceivers);
        }
        contextReceivers.put(receiver, filter);
    }

    /**
     * 取消注册广播
     *
     * @param context  上下文
     * @param receiver 广播接收器
     */
    public static void unregisterReceiver(Context context, BroadcastReceiver receiver) {
        if (context == null || receiver == null) {
            return;
        }

        try {
            context.unregisterReceiver(receiver);
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        }

        // 移除注册信息
        Map<BroadcastReceiver, IntentFilter> contextReceivers = receiverMap.get(context);
        if (contextReceivers != null) {
            contextReceivers.remove(receiver);
            if (contextReceivers.isEmpty()) {
                receiverMap.remove(context);
            }
        }
    }

    /**
     * 发送广播
     *
     * @param context 上下文
     * @param intent  意图
     */
    public static void sendBroadcast(Context context, Intent intent) {
        if (context == null || intent == null) {
            return;
        }
        context.sendBroadcast(intent);
    }

    /**
     * 发送带权限的广播
     *
     * @param context      上下文
     * @param intent       意图
     * @param permission   权限
     */
    public static void sendBroadcast(Context context, Intent intent, String permission) {
        if (context == null || intent == null || permission == null) {
            return;
        }
        context.sendBroadcast(intent, permission);
    }

    /**
     * 发送有序广播
     *
     * @param context      上下文
     * @param intent       意图
     * @param permission   权限
     */
    public static void sendOrderedBroadcast(Context context, Intent intent, String permission) {
        if (context == null || intent == null || permission == null) {
            return;
        }
        context.sendOrderedBroadcast(intent, permission);
    }

    /**
     * 取消注册所有广播
     *
     * @param context 上下文
     */
    public static void unregisterAllReceivers(Context context) {
        if (context == null) {
            return;
        }

        Map<BroadcastReceiver, IntentFilter> contextReceivers = receiverMap.get(context);
        if (contextReceivers != null) {
            for (BroadcastReceiver receiver : contextReceivers.keySet()) {
                try {
                    context.unregisterReceiver(receiver);
                } catch (IllegalArgumentException e) {
                    e.printStackTrace();
                }
            }
            contextReceivers.clear();
            receiverMap.remove(context);
        }
    }
}
  1. 使用示例
    2.1 动态注册广播
// 自定义广播接收器
BroadcastReceiver receiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();
        if (Intent.ACTION_BOOT_COMPLETED.equals(action)) {
            System.out.println("Device boot completed");
        }
    }
};

// 注册广播
IntentFilter filter = new IntentFilter(Intent.ACTION_BOOT_COMPLETED);
BroadcastUtils.registerReceiver(context, receiver, filter);

// 取消注册广播
BroadcastUtils.unregisterReceiver(context, receiver);

2.2 发送广播

// 发送普通广播
Intent intent = new Intent("com.example.CUSTOM_ACTION");
intent.putExtra("key", "value");
BroadcastUtils.sendBroadcast(context, intent);

// 发送带权限的广播
String permission = "com.example.PERMISSION";
BroadcastUtils.sendBroadcast(context, intent, permission);

2.3 发送有序广播

// 发送有序广播
String permission = "com.example.PERMISSION";
BroadcastUtils.sendOrderedBroadcast(context, intent, permission);

2.4 取消注册所有广播

// 取消注册所有广播
BroadcastUtils.unregisterAllReceivers(context);
  1. 高版本适配说明
    3.1 隐式广播限制
    从 Android 8.0(API 26)开始,系统对 隐式广播 进行了限制。如果目标广播是隐式的,需要使用 显式广播 或 JobScheduler 替代。

3.2 后台执行限制
从 Android 9(API 28)开始,系统对后台应用的广播接收器进行了限制。如果应用在后台运行,某些广播可能无法接收。

3.3 动态注册
在高版本中,建议使用 动态注册 的方式注册广播接收器,并在适当的生命周期中取消注册,以避免内存泄漏。

  1. 静态注册广播
    如果需要在应用未启动时接收广播(如开机启动),可以使用 静态注册。以下是一个示例:

4.1 在 AndroidManifest.xml 中声明广播接收器

<receiver android:name=".MyBootReceiver"
    android:exported="true"
    android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
    <intent-filter>
        <action android:name="android.intent.action.BOOT_COMPLETED" />
    </intent-filter>
</receiver>

4.2 实现广播接收器

public class MyBootReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        if (Intent.ACTION_BOOT_COMPLETED.equals(intent.getAction())) {
            System.out.println("Device boot completed");
        }
    }
}
5. 总结

通过封装 Broadcast 工具类,可以:

简化广播的注册和发送。

适配高版本的隐式广播限制。

支持权限控制和有序广播。

避免内存泄漏和重复注册。

建议将工具类封装到一个独立的模块中,方便在项目中复用。根据实际需求,可以进一步扩展功能,如支持跨进程广播、自定义权限等。


http://www.kler.cn/a/580667.html

相关文章:

  • 基于深度学习的中文文本情感分析系统
  • 我与DeepSeek读《大型网站技术架构》(8)- 网站应用攻击与防御
  • 智能化时代,美容院如何借助科技实现高效运营?
  • 行为级建模
  • React框架连续解构赋值详细解读
  • Nest.js全栈开发终极实践:TypeORM+微服务+Docker构建高可用企业级应用
  • LeetCode --- 439周赛
  • HarmonyOS Next 中的状态管理
  • 推理LLMs发展综述:从GPT到DeepSeek
  • 手机号实名认证接口:数字时代的安全与便捷保障
  • 【IPFS应用开发】IPFS播放器-上传助手
  • 深度学习实战车辆目标跟踪与计数
  • 【网络协议详解】——MPLS LDP技术(学习笔记)
  • MySQL数据库操作
  • HarmonyOS NEXT开发实战:DevEco AI辅助编程工具(CodeGenie)的使用
  • QT系列教程(14) QT 按键事件
  • 【病毒分析】熊猫烧香病毒分析及其查杀修复
  • 【自学笔记】Rust语言基础知识点总览-持续更新
  • 两会聚焦科技金融创新,赛逸展2025成重要实践平台
  • 关于前后端整合和打包成exe文件的个人的总结和思考