在Android8.0上突破隐式广播的限制
Android O对隐式广播进行了限制, 其限制链接说明: https://developer.android.com/about/versions/oreo/background

上面所说即:若App的TargetSDK达到了26, 我们正常静态注册的广播就没有使用了。能使用的仅有以下豁免的Broadcast, 包括我们自己正常发广播,假如不指定包名, 静态注册的也是收不到的。PS:动态注册是没有影响的
https://developer.android.com/guide/components/broadcast-exceptions
在我们收不到广播的时候,系统会有如下打印,即这个后端的广播接收器不会被执行
04-21 04:12:27.513 2431 4821 W BroadcastQueue: Background execution not allowed:******************
如何应对这一限制
知道了上面的限制后,我们正常的应对方式为
- 能动态注册,就不静态注册
- 假如肯定要静态注册, 发送的时候指定包名,即发送显式广播
- 假如要接收系统广播,而对应的广播在Android8.0中无法被接收,那么只能暂时把App的targetSdkVersion改为25或者以下,但这招已经不顶使用了,工信部要求targetSDK必需26以上
假如我们不想发显式广播(由于我们不知道有谁要收广播),对方又不能动态注册,只能静态注册(许多应使用希望是被动唤醒),我们应该怎样办呢?
我们看上面的异常:
04-21 04:12:27.513 2431 4821 W BroadcastQueue: Background execution not allowed:******************
这行log是哪边打印的呢?
我们去搜索一下:http://androidxref.com/
其代码在:frameworks/base/services/core/java/com/android/server/am/BroadcastQueue.java#1275
代码里面有个skip变量是使用来标志能否跳过的,很显然1275行打印出来了,skip为true了那就, 我们不希望这个判断能够进去。
那么如合让判断不进去呢?看下面代码。
1267 } else if (((r.intent.getFlags()&Intent.FLAG_RECEIVER_EXCLUDE_BACKGROUND) != 0)1268 || (r.intent.getComponent() == null1269 && r.intent.getPackage() == null1270 && ((r.intent.getFlags()1271 & Intent.FLAG_RECEIVER_INCLUDE_BACKGROUND) == 0)1272 && !isSignaturePerm(r.requiredPermissions))) {1273 mService.addBackgroundCheckViolationLocked(r.intent.getAction(),1274 component.getPackageName());1275 Slog.w(TAG, "Background execution not allowed: receiving "1276 + r.intent + " to "1277 + component.flattenToShortString());1278 skip = true;1279 }- 有这么个判断
r.intent.getFlags()&Intent.FLAG_RECEIVER_EXCLUDE_BACKGROUNDintent中携带了EXCLUDE_BACKGROUND标志位一定进去,很显然我们正常都不带,只有希望后端收不到的时候才会带。 r.intent.getComponent() == null, 这个一定不会为null的。为null是必需跳过r.intent.getPackage() == null, 若包名为空,那一定也跳过r.intent.getFlags() & Intent.FLAG_RECEIVER_INCLUDE_BACKGROUND) == 0不能带有FLAG_RECEIVER_INCLUDE_BACKGROUND这个标志位,若带了,那就进不去了,这不就是我们希望的结果么。
那么方案有了,携带 FLAG_RECEIVER_INCLUDE_BACKGROUND 这个标志位。我们发现在AS中用Intent是找不到这个标志位的,应该是hide了,没有被编译进SDK。
看一下,果然,那么我们直接带硬编码就可。
/** * If set, the broadcast will always go to manifest receivers in background (cached * or not running) apps, regardless of whether that would be done by default. By * default they will only receive broadcasts if the broadcast has specified an * explicit component or package name. * * NOTE: dumpstate uses this flag numerically, so when its value is changed * the broadcast code there must also be changed to match. * * @hide */public static final int FLAG_RECEIVER_INCLUDE_BACKGROUND = 0x01000000;因而得出结论:发送广播的时候携带intent.addFlags(0x01000000); 即能让广播突破隐式广播限制。
谢谢大家的阅读。
本文作者:Anderson/Jerey_Jobs
博客地址 : http://jerey.cn/
简书地址 : Anderson大码渣
github地址 : Jerey-Jobs
上一篇 目录 已是最后
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 在Android8.0上突破隐式广播的限制