分享好友 最新动态首页 最新动态分类 切换频道
2024年Java最全BroadcastReceiver源码分析,手把手教你写
2024-12-26 21:28

最后

做任何事情都要用心,要非常关注细节。看起来不起眼的、繁琐的工作做透了会有意想不到的价值。
当然要想成为一个技术大牛也需要一定的思想格局,思想决定未来你要往哪个方向去走, 建议多看一些人生规划方面的书籍,多学习名人的思想格局,未来你的路会走的更远。

更多的技术点思维导图我已经做了一个整理,涵盖了当下互联网最流行99%的技术点,在这里我将这份导图分享出来,以及为金九银十准备的一整套面试体系,上到集合,下到分布式微服务

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

}

2.3 AMS发送到队列处理广播


queue.scheduleBroadcastsLocked();会发送BROADCAST_INTENT_MSG,调用processNextBroadcast(true);

BroadcastQueue.java

final void processNextBroadcast(boolean fromMsg) {

synchronized (mService) {

processNextBroadcastLocked(fromMsg, false);

}

}

final void processNextBroadcastLocked(boolean fromMsg, boolean skipOomAdj){

while (mParallelBroadcasts.size() > 0) {

r = mParallelBroadcasts.remove(0);

final int N = r.receivers.size();

//依次取出分发

for (int i=0; i<N; i++) {

Object target = r.receivers.get(i);

//调用

deliverToRegisteredReceiverLocked(r, (BroadcastFilter)target, false, i);

}

}

}

private void deliverToRegisteredReceiverLocked(BroadcastRecord r,

BroadcastFilter filter, boolean ordered, int index) {

··· //分发广播到客户端

performReceiveLocked(filter.receiverList.app, filter.receiverList.receiver,

new Intent(r.intent), r.resultCode, r.resultData,

r.resultExtras, r.ordered, r.initialSticky, r.userId);

····

}

void performReceiveLocked(ProcessRecord app, IIntentReceiver receiver,

Intent intent, int resultCode, String data, Bundle extras,

boolean ordered, boolean sticky, int sendingUser) throws RemoteException {

// Send the intent to the receiver asynchronously using one-way binder calls.

//判断应是否为空

if (app != null) {

if (app.thread != null) {

//通知客户端分发广播,Binder采用 oneway 异步 应用端会自动串行化

app.thread.scheduleRegisteredReceiver(receiver, intent, resultCode,

data, extras, ordered, sticky, sendingUser, app.repProcState);

}

} else {

receiver.performReceive(intent, resultCode, data, extras, ordered,

sticky, sendingUser);

}

}

2.4 客户端处理receiver


AMS端的app.thread.scheduleRegisteredReceiver(),会调用到注册时注册的InnerReceiver的performReceive方法

ActivityThread.java

public void scheduleRegisteredReceiver(IIntentReceiver receiver, Intent intent,

int resultCode, String dataStr, Bundle extras, boolean ordered,

boolean sticky, int sendingUser, int processState) throws RemoteException {

updateProcessState(processState, false);

//1

receiver.performReceive(intent, resultCode, dataStr, extras, ordered,

sticky, sendingUser);

}

InnerReceiver.java

public void performReceive(Intent intent, int resultCode, String data,

Bundle extras, boolean ordered, boolean sticky, int sendingUser) {

final LoadedApk.ReceiverDispatcher rd;

···

if (rd != null) {

//调用LoadedApk.ReceiverDispatcher的performReceive

rd.performReceive(intent, resultCode, data, extras,

ordered, sticky, sendingUser);

···

}

}

···

LoadedApk.ReceiverDispatcher.java

public void performReceive(Intent intent, int resultCode, String data,

Bundle extras, boolean ordered, boolean sticky, int sendingUser) {

final Args args = new Args(intent, resultCode, data, extras, ordered,

sticky, sendingUser);

···

//向ActivityThread发送一个Runnable

if (intent == null || !mActivityThread.post(args.getRunnable())) {

}

}

Args.java

public final Runnable getRunnable() {

return () -> {

final BroadcastReceiver receiver = mReceiver;

final boolean ordered = mOrdered;

=

final IActivityManager mgr = ActivityManager.getService();

···

receiver.setPendingResult(this);

//这里触发了BoradcastReciever回调

receiver.onReceive(mContext, intent);

···

};

}

三、动态注册解析总结

=========================================================================

1、首先动态注册BroadcastReceiver到AMS,将他们存储在一个Map中,Map的key是IIntentReceiver用于回调注册端,value是一个ReceiverList

2、发送这发送Intent到AMS,筛选出匹配的ReceiverList,遍历通知注册端

四、BroadcastReceiver静态注册接收

=======================================================================================

4.1 PMS解析清单文件获取注册信息


PackageParser.java

private boolean parseBaseApplication(Package owner, Resources res,

XmlResourceParser parser, int flags, String[] outError){

else if (tagName.equals(“receiver”)) {

Activity a = parseActivity(owner, res, parser, flags, outError, cachedArgs,

true, false);

owner.receivers.add(a);

}

}

1、将receiver解析成一个Activity(这个Activity表示的是一个组件)对象,添加到receivers的列表中,receivers是一个ArrayList

五、普通广播发送后的静态接收

============================================================================

5.1 调用Context的sendBroadcast


最后会调用到

ContextImpl.java

public void sendBroadcast(Intent intent) {

warnIfCallingFromSystemProcess();

String resolvedType = intent.resolveTypeIfNeeded(getContentResolver());

intent.prepareToLeaveProcess(this);

ActivityManager.getService().broadcastIntent(

mMainThread.getApplicationThread(), intent, resolvedType, null,

Activity.RESULT_OK, null, null, null, AppOpsManager.OP_NONE, null, false, false,

getUserId());

}

5.2 AMS处理broadcastIntent


ActivityManagerService.java

public final int broadcastIntent(IApplicationThread caller,

Intent intent, String resolvedType, IIntentReceiver resultTo,

int resultCode, String resultData, Bundle resultExtras,

String[] requiredPermissions, int appOp, Bundle bOptions,

boolean serialized, boolean sticky, int userId) {

···

int res = broadcastIntentLocked(callerApp,

callerApp != null ? callerApp.info.packageName : null,

intent, resolvedType, resultTo, resultCode, resultData, resultExtras,

requiredPermissions, appOp, bOptions, serialized, sticky,

callingPid, callingUid, userId);

···

}

}

final int broadcastIntentLocked(ProcessRecord callerApp,

String callerPackage, Intent intent, String resolvedType,

IIntentReceiver resultTo, int resultCode, String resultData,

Bundle resultExtras, String[] requiredPermissions, int appOp, Bundle bOptions,

boolean ordered, boolean sticky, int callingPid, int callingUid, int userId){

···

//获取接收静态广播

receivers = collectReceiverComponents(intent, resolvedType, callingUid, users);

//通过Intent查找能接受的动态广播

registeredReceivers = mReceiverResolver.queryIntent(intent,

resolvedType, false /defaultOnly/, userId);

int NR = registeredReceivers != null ? registeredReceivers.size() : 0;

if (!ordered && NR > 0) {

//处理动态广播,添加到并行分发队列,OneWay发送binder处理是串行的

}

//给没有处理完的动态receiver(order是true,跟静态receiver合并到一起

if ((receivers != null && receivers.size() > 0){

//处理剩下的receiver。加到串行分发队列

//获取queue

BroadcastQueue queue = broadcastQueueForIntent(intent);

BroadcastRecord r = new BroadcastRecord(queue, intent, callerApp,

callerPackage, callingPid, callingUid, callerInstantApp, resolvedType,

requiredPermissions, appOp, brOptions, receivers, resultTo, resultCode,

resultData, resultExtras, ordered, sticky, false, userId);

//添加到串行分发queue

queue.enqueueOrderedBroadcastLocked®;

//处理分发

queue.scheduleBroadcastsLocked();

}

return ActivityManager.BROADCAST_SUCCESS;

}

5.3 AMS发送到队列处理广播


queue.scheduleBroadcastsLocked();会发送BROADCAST_INTENT_MSG,调用processNextBroadcast(true);

BroadcastQueue.java

final void processNextBroadcast(boolean fromMsg) {

synchronized (mService) {

processNextBroadcastLocked(fromMsg, false);

}

}

final void processNextBroadcastLocked(boolean fromMsg, boolean skipOomAdj){

while (mParallelBroadcasts.size() > 0) {

r = mParallelBroadcasts.remove(0);

final int N = r.receivers.size()

for (int i=0; i<N; i++) {

Object target = r.receivers.get(i);

//先并行分发,然后接下来分发串行广播

deliverToRegisteredReceiverLocked(r, (BroadcastFilter)target, false, i);

}

}

//如果有pedding广播,先直接返回,这个广播在等待应用进程启动

//如果当前广播分发超时了,废弃这个广播,处理下一个广播

broadcastTimeoutLocked(false);

//如果没有超时,并且在分发中,就先返回。什么也不做

//如果当前的广播已经分发完一个receiver,就继续分发下一个receiver

//如果这个receiver是动态注册的receiver就直接分发

//如果这个receiver是静态注册的receiver,先看进程启动没有

//如果进程启动了,就直接分发

processCurBroadcastLocked(r, app, skipOomAdj);

//没启动的话就先启动进程,然后给广播标记为pedding

//进程启动后attachApplication时继续处理这个pending的广播

}

BroadcastQueue.java

//处理超时

final void broadcastTimeoutLocked(boolean fromMsg) {

BroadcastRecord r = mOrderedBroadcasts.get(0);

Object curReceiver = r.receivers.get(r.nextReceiver-1);

//找到当前分发receiver对应的进程

if (mPendingBroadcast == r) {

mPendingBroadcast = null;

}

// Move on to the next receiver.

finishReceiverLocked(r, r.resultCode, r.resultData,

r.resultExtras, r.resultAbort, false);

//发送消息处理下一个receiver

scheduleBroadcastsLocked();

if (!debugging && anrMessage != null) {

// Post the ANR to the handler since we do not want to process ANRs while

// potentially holding our lock.

//超时显示ANR

mHandler.post(new AppNotResponding(app, anrMessage));

}

}

BroadcastQueue.java

//接收进程存在,通知接收进程

private final void processCurBroadcastLocked(BroadcastRecord r,

ProcessRecord app, boolean skipOomAdj) throws RemoteException {

···

app.thread.scheduleReceiver(new Intent(r.intent), r.curReceiver,

mService.compatibilityInfoForPackageLocked(r.curReceiver.applicationInfo),

r.resultCode, r.resultData, r.resultExtras, r.ordered, r.userId,

app.repProcState);

···

}

5.4 接收进程接收广播


5.4.1 进程存在直接处理

ActivityThread.java

public final void scheduleReceiver(Intent intent, ActivityInfo info,

CompatibilityInfo compatInfo, int resultCode, String data, Bundle extras,

boolean sync, int sendingUser, int processState) {

updateProcessState(processState, false);

ReceiverData r = new ReceiverData(intent, resultCode, data, extras,

sync, false, mAppThread.asBinder(), sendingUser);

r.info = info;

r.compatInfo = compatInfo;

sendMessage(H.RECEIVER, r);

}

private void handleReceiver(ReceiverData data) {

sCurrentBroadcastIntent.set(data.intent);

receiver.setPendingResult(data);

//执行BroadcastReceiver

//依据Application的Context创建的ContextWarpper,目的是不允许BroadcastReceiver在注册接收器,也不允许启动服务

receiver.onReceive(context.getReceiverRestrictedContext(),

data.intent);

if (receiver.getPendingResult() != null) {

//通知AMS分发结束

data.finish();

}

}

BroadcastReceiver.java

public final void finish() {

if (mType == TYPE_COMPONENT) {

final IActivityManager mgr = ActivityManager.getService();

if (QueuedWork.hasPendingWork()) {

QueuedWork.queue(new Runnable() {

@Override public void run() {

if (ActivityThread.DEBUG_BROADCAST) Slog.i(ActivityThread.TAG,

"Finishing broadcast after work to component " + mToken);

sendFinished(mgr);

}

}, false);

} else {

sendFinished(mgr);

}

} else if (mOrderedHint && mType != TYPE_UNREGISTERED) {

final IActivityManager mgr = ActivityManager.getService();

sendFinished(mgr);

}

}

public void sendFinished(IActivityManager am) {

//通知AMS分发结束

看完美团、字节、腾讯这三家的面试问题,是不是感觉问的特别多,可能咱们又得开启面试造火箭、工作拧螺丝的模式去准备下一次的面试了。

开篇有提及我可是足足背下了1000道题目,多少还是有点用的呢,我看了下,上面这些问题大部分都能从我背的题里找到的,所以今天给大家分享一下互联网工程师必备的面试1000题

注意不论是我说的互联网面试1000题,还是后面提及的算法与数据结构、设计模式以及更多的Java学习笔记等,皆可分享给各位朋友

互联网工程师必备的面试1000题

而且从上面三家来看算法与数据结构是必备不可少的,因此我建议大家可以去刷刷这本左程云大佬著作的《程序员代码面试指南 IT名企算法与数据结构题目最优解》,里面近200道真实出现过的经典代码面试题

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

final IActivityManager mgr = ActivityManager.getService();

sendFinished(mgr);

}

}

public void sendFinished(IActivityManager am) {

//通知AMS分发结束

看完美团、字节、腾讯这三家的面试问题,是不是感觉问的特别多,可能咱们又得开启面试造火箭、工作拧螺丝的模式去准备下一次的面试了。

开篇有提及我可是足足背下了1000道题目,多少还是有点用的呢,我看了下,上面这些问题大部分都能从我背的题里找到的,所以今天给大家分享一下互联网工程师必备的面试1000题

注意不论是我说的互联网面试1000题,还是后面提及的算法与数据结构、设计模式以及更多的Java学习笔记等,皆可分享给各位朋友

[外链图片转存中…(img-gReWvylz-1714849681530)]

互联网工程师必备的面试1000题

而且从上面三家来看算法与数据结构是必备不可少的,因此我建议大家可以去刷刷这本左程云大佬著作的《程序员代码面试指南 IT名企算法与数据结构题目最优解》,里面近200道真实出现过的经典代码面试题

[外链图片转存中…(img-I1bS0Tjj-1714849681530)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

最新文章
Vol.293 很多年坚持做一件事是种什么体验?(本期之后,开启不定期更新模式)
有一个事跟听友们汇报:《音乐故事》即将转入战略修养期。放心,不是不做了。 是开启不定期更新模式。时光荏苒,音乐故事从2014年初开始到现在,已经陪伴大家五年半的时间了。这期间,无论是过年、旅游、生娃,我们俩都没有停止过更新,虽
竞价广告代运营服务包括哪些内容?全方位了解代运营服务范畴!
竞价广告代运营服务,简而言之,就是由专业的团队代替广告主进行广告投放的全过程管理。这种服务旨在通过精准的策略制定和执行,帮助广告主在竞争激烈的市场中脱颖而出,实现广告效果的最大化。关键词是竞价广告的核心。代运营团队会深入分
谷歌SEO服务,需要收费吗?
在如今这个数字化时代,搜索引擎优化(SEO)已成为企业提升网站流量、提高品牌曝光度的关键手段。作为一名长期在SEO领域摸爬滚打的从业者,我深知谷歌SEO服务的重要性。面对激烈的市场竞争,企业对于SEO服务的需求日益增加,而关于谷歌SEO
石头在线翻译插件
翻译工具是一百元/月,六百包年。可以在线测试效果,效果优于老版的。感觉还算满意就加小北。近期群里几个小伙伴请求我带着一起做做海外市场,于是建个平台并提供一些工具和基础技术、知识分享,方便大家避坑。小北的主要精力主要在做站方
回放丨“奋进新征程·广西改革这十年”新闻发布会
  广西壮族自治区党委宣传部于2022年9月28日(星期三)10时,在广西新闻发布厅举行“奋进新征程·广西改革这十年”新闻发布会,邀请自治区党委改革办专职副主任何运安,自治区党委政法委副书记罗棋权,自治区发展改革委党组成员樊一江,
郑州 SEO 外链建设终极指南:提升网站排名的策略
外链建设对于提高搜索引擎优化 (SEO) 排名至关重要。它向搜索引擎表明您的网站具有可信度和权威性,这在确定网站在搜索结果页面 (SERP) 中的排名时至关重要。对于郑州当地企业而言,拥有一个战略性的外链建设计划对于在竞争激烈的市场中脱
百度推广怎么调价【百度推广怎么提高点击率】
本文目录导读:市场行情分析百度推广调价的重要性百度推广调价的方法和技巧注意事项和风险防范在当今数字化营销的时代,百度推广作为一种重要的网络推广手段,对于企业和广告主来说具有不可忽视的作用,而百度推广的调价策略则是影响推广效
金山词霸2008下载 [金山新年大礼:金山词霸个人版终身免费公测申请]
这次是出乎X-Force的预料了,金山公司在临近新年的时候给了我们一个惊喜,他们宣布了金山词霸2008个人版从今起完全免费啦!而且还承诺是终身免费呢。之前我还在想有了有道桌面词典、灵格斯词霸等优秀且免费词典软件之后,金山的份额肯定会
抖音短视频跳转链接怎么生成的呢?
市场营销获客工具【爱短链】全域跳转工具,打通私域引流关键一步:轻松将公域流量,引导至私域,通过链接点击直达,方便快捷,无风险提示跳转,助力企业/商家高速获客》》点此生成链接 我知道有三个环境可以从抖音跳转到微信小程序,即短视
相关文章
推荐文章
发表评论
0评