BroadcastReceiver官方概览(3)

系统广播所发生的更改

==========

随着 Android 平台的发展,它会不定期地更改系统广播的行为方式。如果您的应用以 Android 7.0(API 级别 24)或更高版本为目标平台,或者安装在搭载 Android 7.0 或更高版本的设备上,请注意以下更改。

Android 9

从 Android 9(API 级别 28)开始,NETWORK_STATE_CHANGED_ACTION 广播不再接收有关用户位置或个人身份数据的信息。

此外,如果您的应用安装在搭载 Android 9 或更高版本的设备上,则通过 WLAN 接收的系统广播不包含 SSID、BSSID、连接信息或扫描结果。要获取这些信息,请调用 getConnectionInfo()

Android 8.0

从 Android 8.0(API 级别 26)开始,系统对清单声明的接收器施加了额外的限制。

如果您的应用以 Android 8.0 或更高版本为目标平台,那么对于大多数隐式广播(没有明确针对您的应用的广播),您不能使用清单来声明接收器。当用户正在活跃地使用您的应用时,您仍可使用上下文注册的接收器

Android 7.0

Android 7.0(API 级别 24)及更高版本不发送以下系统广播:

  • [ACTION_NEW_PICTURE]( )

  • [ACTION_NEW_VIDEO]( )

此外,以 Android 7.0 及更高版本为目标平台的应用必须使用 [registerReceiver(BroadcastReceiver, IntentFilter)]( ) 注册 [CONNECTIVITY_ACTION]( ) 广播。无法在清单中声明接收器。

接收广播

====

应用可以通过两种方式接收广播:清单声明的接收器和上下文注册的接收器。

清单声明的接收器


如果您在清单中声明广播接收器,系统会在广播发出后启动您的应用(如果应用尚未运行)。

注意:如果您的应用以 API 级别 26 或更高级别的平台版本为目标,则不能使用清单为隐式广播(没有明确针对您的应用的广播)声明接收器,但一些不受此限制的隐式广播除外。在大多数情况下,您可以使用调度作业来代替。

要在清单中声明广播接收器,请执行以下步骤:

1、在应用清单中指定 <receiver> 元素。

Intent 过滤器指定您的接收器所订阅的广播操作。

2、创建 [BroadcastReceiver]( ) 子类并实现 [onReceive(Context, Intent)]( )。以下示例中的广播接收器会记录并显示广播的内容:

public class MyBroadcastReceiver extends BroadcastReceiver {

private static final String TAG = “MyBroadcastReceiver”;

@Override

public void onReceive(Context context, Intent intent) {

StringBuilder sb = new StringBuilder();

sb.append("Action: " + intent.getAction() + “\n”);

sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + “\n”);

String log = sb.toString();

Log.d(TAG, log);

Toast.makeText(context, log, Toast.LENGTH_LONG).show();

}

}

系统软件包管理器会在应用安装时注册接收器。然后,该接收器会成为应用的一个独立入口点,这意味着如果应用当前未运行,系统可以启动应用并发送广播。

系统会创建新的 [BroadcastReceiver]( ) 组件对象来处理它接收到的每个广播。此对象仅在调用 [onReceive(Context, Intent)]( ) 期间有效。一旦从此方法返回代码,系统便会认为该组件不再活跃。

上下文注册的接收器


要使用上下文注册接收器,请执行以下步骤:

1、创建 [BroadcastReceiver]( ) 的实例。

BroadcastReceiver br = new MyBroadcastReceiver();

2、创建 [IntentFilter]( ) 并调用 [registerReceiver(BroadcastReceiver, IntentFilter)]( ) 来注册接收器:

IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);

filter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED);

this.registerReceiver(br, filter);

**注意:**要注册本地广播,请调用 [LocalBroadcastManager.registerReceiver(BroadcastReceiver, IntentFilter)]( )

只要注册上下文有效,上下文注册的接收器就会接收广播。例如,如果您在 [Activity]( ) 上下文中注册,只要 Activity 没有被销毁,您就会收到广播。如果您在应用上下文中注册,只要应用在运行,您就会收到广播。

3、要停止接收广播,请调用 [unregisterReceiver(android.content.BroadcastReceiver)]( )。当您不再需要接收器或上下文不再有效时,请务必注销接收器。

请注意注册和注销接收器的位置,比方说,如果您使用 Activity 上下文在 [onCreate(Bundle)]( ) 中注册接收器,则应在 [onDestroy()]( ) 中注销,以防接收器从 Activity 上下文中泄露出去。如果您在 [onResume()]( ) 中注册接收器,则应在 [onPause()]( ) 中注销,以防多次注册接收器(如果您不想在暂停时接收广播,这样可以减少不必要的系统开销)。请勿在 [onSaveInstanceState(Bundle)]( ) 中注销,因为如果用户在历史记录堆栈中后退,则不会调用此方法。

对进程状态的影响

========

[BroadcastReceiver]( ) 的状态(无论它是否在运行)会影响其所在进程的状态,而其所在进程的状态又会影响它被系统终结的可能性。例如,当进程执行接收器(即当前在运行其 [onReceive()]( ) 方法中的代码)时,它被认为是前台进程。除非遇到极大的内存压力,否则系统会保持该进程运行。

但是,一旦从 onReceive() 返回代码,BroadcastReceiver 就不再活跃。接收器的宿主进程变得与在其中运行的其他应用组件一样重要。如果该进程仅托管清单声明的接收器(这对于用户从未与之互动或最近没有与之互动的应用很常见),则从 onReceive() 返回时,系统会将其进程视为低优先级进程,并可能会将其终止,以便将资源提供给其他更重要的进程使用。

因此,您不应从广播接收器启动长时间运行的后台线程。onReceive() 完成后,系统可以随时终止进程来回收内存,在此过程中,也会终止进程中运行的派生线程。要避免这种情况,您应该调用 [goAsync()]( )(如果您希望在后台线程中多花一点时间来处理广播)或者使用 [JobScheduler]( ) 从接收器调度 [JobService]( ),这样系统就会知道该进程将继续活跃地工作。如需了解详情,请参阅进程和应用生命周期

以下代码段展示了一个 [BroadcastReceiver]( ),它使用 [goAsync()]( ) 来标记它在 onReceive() 完成后需要更多时间才能完成。如果您希望在 onReceive() 中完成的工作很长,足以导致界面线程丢帧 (>16ms),则这种做法非常有用,这使它尤其适用于后台线程。

public class MyBroadcastReceiver extends BroadcastReceiver {

private static final String TAG = “MyBroadcastReceiver”;

@Override

public void onReceive(Context context, Intent intent) {

final PendingResult pendingResult = goAsync();

Task asyncTask = new Task(pendingResult, intent);

asyncTask.execute();

}

private static class Task extends AsyncTask<String, Integer, String> {

private final PendingResult pendingResult;

private final Intent intent;

private Task(PendingResult pendingResult, Intent intent) {

this.pendingResult = pendingResult;

this.intent = intent;

}

@Override

protected String doInBackground(String… strings) {

StringBuilder sb = new StringBuilder();

sb.append("Action: " + intent.getAction() + “\n”);

sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + “\n”);

String log = sb.toString();

Log.d(TAG, log);

return log;

}

@Override

protected void onPostExecute(String s) {

super.onPostExecute(s);

// Must call finish() so the BroadcastReceiver can be recycled.

pendingResult.finish();

}

}

}

发送广播

====

Android 为应用提供三种方式来发送广播:

  • [sendOrderedBroadcast(Intent, String)]( ) 方法一次向一个接收器发送广播。当接收器逐个顺序执行时,接收器可以向下传递结果,也可以完全中止广播,使其不再传递给其他接收器。接收器的运行顺序可以通过匹配的 intent-filter 的 android:priority 属性来控制;具有相同优先级的接收器将按随机顺序运行。

  • [sendBroadcast(Intent)]( ) 方法会按随机的顺序向所有接收器发送广播。这称为常规广播。这种方法效率更高,但也意味着接收器无法从其他接收器读取结果,无法传递从广播中收到的数据,也无法中止广播。

  • [LocalBroadcastManager.sendBroadcast]( ) 方法会将广播发送给与发送器位于同一应用中的接收器。如果您不需要跨应用发送广播,请使用本地广播。这种实现方法的效率更高(无需进行进程间通信),而且您无需担心其他应用在收发您的广播时带来的任何安全问题。

以下代码段展示了如何通过创建 Intent 并调用 [sendBroadcast(Intent)]( ) 来发送广播。

Intent intent = new Intent();

intent.setAction(“com.example.broadcast.MY_NOTIFICATION”);

intent.putExtra(“data”,“Notice me senpai!”);

sendBroadcast(intent);

广播消息封装在 [Intent]( ) 对象中。Intent 的操作字符串必须提供应用的 Java 软件包名称语法,并唯一标识广播事件。您可以使用 [putExtra(String, Bundle)]( ) 向 intent 附加其他信息。您也可以对 intent 调用 [setPackage(String)]( ),将广播限定到同一组织中的一组应用。

注意:虽然 intent 既用于发送广播,也用于通过 [startActivity(Intent)]( ) 启动 Activity,但这两种操作是完全无关的。广播接收器无法查看或捕获用于启动 Activity 的 intent;同样,当您广播 intent 时,也无法找到或启动 Activity。

通过权限限制广播

========

您可以通过权限将广播限定到拥有特定权限的一组应用。您可以对广播的发送器或接收器施加限制。

带权限的发送


当您调用 [sendBroadcast(Intent, String)]( ) 或 [sendOrderedBroadcast(Intent, String, BroadcastReceiver, Handler, int, String, Bundle)]( ) 时,可以指定权限参数。接收器若要接收此广播,则必须通过其清单中的 标记请求该权限(如果存在危险,则会被授予该权限)。例如,以下代码会发送广播:

sendBroadcast(new Intent(“com.example.NOTIFY”), Manifest.permission.SEND_SMS);

要接收此广播,接收方应用必须请求如下权限:

您可以指定现有的系统权限(如 [SEND_SMS]( )),也可以使用 <permission> 元素定义自定义权限。有关权限和安全性的一般信息,请参阅系统权限

**注意:**自定义权限将在安装应用时注册。定义自定义权限的应用必须在使用自定义权限的应用之前安装。

带权限的接收


如果您在注册广播接收器时指定了权限参数(通过 [registerReceiver(BroadcastReceiver, IntentFilter, String, Handler)]( ) 或清单中的 <receiver> 标记指定),则广播方必须通过其清单中的 <uses-permission> 标记请求该权限(如果存在危险,则会被授予该权限),才能向该接收器发送 Intent。

例如,假设您的接收方应用具有如下所示的清单声明的接收器:

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
mg-EuFk05O5-1715503967302)]

[外链图片转存中…(img-RIEsBDWi-1715503967303)]

[外链图片转存中…(img-Mbcn3GQC-1715503967304)]

[外链图片转存中…(img-s64G8hqc-1715503967305)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 14
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值