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

Android --- 在AIDL进程间通信中,为什么使用RemoteCallbackList 代替 ArrayList?

1.RemoteCallbackList vs ArrayList
RemoteCallbackList 是一个特殊的 List,它用来管理跨进程的回调,特别是当回调对象是在不同进程中时。它在 AIDL(Android Interface Definition Language)通信中常常用来处理跨进程的通信。
ArrayList 是普通的 Java 集合类,它通常不适合在 AIDL 中使用,尤其是在进程间通信中。
RemoteCallbackList 的优势:

RemoteCallbackList 可以有效地管理跨进程通信中的回调对象,并且能够处理回调对象因进程崩溃(如 binder dead)而被移除的问题。
当一个客户端(比如进程 C)崩溃或与服务的连接断开时,RemoteCallbackList 会自动移除这个客户端的回调,避免了因为失效的回调导致后续客户端的通信被阻塞的问题。
ArrayList 的问题:

如果你使用 ArrayList 存储跨进程的回调对象,当某个进程发生 binder dead(即进程崩溃或连接断开)时,ArrayList 无法自动移除失效的回调,这可能导致内存泄漏或者后续的通信失败。
在进程间通信时,如果使用 ArrayList 存储 AIDL 回调对象,可能会导致一个进程崩溃后,ArrayList 中的其他元素(即其他连接到服务的客户端)也会受到影响,从而阻塞正常的通信。

2. binder dead 问题
binder dead 是指 AIDL 连接的进程发生崩溃或者断开连接时,AIDL 的 Binder 机制无法再和该进程通信,导致调用失败。
如果多个进程(比如 A、B、C、D)通过 AIDL 连接同一个服务(比如服务 E),在默认情况下,如果某个进程(例如 C)发生了 binder dead,该进程将从回调列表中被移除。
如果使用的是普通的 ArrayList,则这个 binder dead 的事件可能不会及时被处理,导致后续的进程(如 D)也会受到阻塞。因为 ArrayList 不能自动移除无效的回调对象,因此服务 E 可能会试图与已经失效的进程通信,导致阻塞。

3. 具体的例子说明
假设有 4 个进程 A、B、C、D,且它们都通过 AIDL 连接到了同一个服务 E。

正常情况:服务 E 会为每个进程(A、B、C、D)注册回调。服务 E 可以向这些进程发送回调消息。

C 崩溃的情况:如果进程 C 崩溃(即发生 binder dead),那么如果使用的是 ArrayList 存储回调对象,服务 E 仍然会尝试通过 ArrayList 向 C 发送回调信息。这会导致一些问题,例如:

服务 E 会持续尝试与 C 通信,浪费资源。
后续的进程(如 D)可能会因为某种同步机制或等待 C 的回调而遭遇阻塞。
这就是为什么使用 ArrayList 在 AIDL 中会带来潜在问题,特别是在处理进程崩溃时。

使用 RemoteCallbackList:如果使用 RemoteCallbackList,当 C 进程崩溃时,RemoteCallbackList 会自动移除 C 的回调,避免了对 C 进程的无效通信。这样,服务 E 可以继续正常向 A、B、D 等其他进程发送回调信息,不会被 C 的崩溃所阻塞。

4. 总结
RemoteCallbackList 是专门为跨进程通信设计的,它能够有效地处理进程崩溃的情况,避免崩溃的进程阻塞其他进程的通信。
ArrayList 在 AIDL 中不适用于存储跨进程的回调,因为它无法自动移除已经失效的回调对象,可能导致 binder dead 事件后,其他进程的通信被阻塞。
在 AIDL 中使用 RemoteCallbackList 是一种更安全和高效的方式,尤其是当多个进程需要通过同一个服务进行通信时。

以上一句话概括:RemoteCallbackList 代替ArrayList
它经常使用于AIDL进程间通信,如果AIDL使用ArrayList,可能会导致binderdead 不能恢复
比如 ABCD 4个进程都通过AIDL连接了一个Service E,C如果binderdead了就会阻塞之后的D与E的通信

http://www.lryc.cn/news/510673.html

相关文章:

  • ADC(二):外部触发
  • 数仓开发那些事(8)
  • 【CSS in Depth 2 精译_096】16.4:CSS 中的三维变换 + 16.5:本章小结
  • 【连续学习之ResCL算法】2020年AAAI会议论文:Residual continual learning
  • 【zookeeper核心源码解析】第二课:俯瞰QuorumPeer启动核心流程,实现选举关键流程
  • 数据流图和流程图的区别
  • 关于内网服务器依托可上网电脑实现访问互联网
  • 期权懂|期权入门知识:如何选择期权合约?
  • 如何用gpt来分析链接里面的内容(比如分析论文链接)和分析包含多个文件中的一块代码
  • Bash 脚本教程
  • Pinia最简单使用(vite+vue3)
  • 计算机网络——期末复习(4)协议或技术汇总、思维导图
  • Microsoft word@【标题样式】应用不生效(主要表现为在导航窗格不显示)
  • 轮播图带详情插件、uniApp插件
  • 云计算时代携程的网络架构变迁
  • USB 状态机及状态转换
  • Go C编程 第6课 无人机 --- 计算旋转角
  • C++-----图
  • mysql 数据库迁移到达梦数据库
  • 【记录】使用R2 CDN替换本地项目图片以加速图片加载
  • 12.13[java exp4][debug]nginx 500,究极未解之谜,出自重启,解决自重启,迷???
  • Disruptor 高性能环形消息框架
  • Python列表(二)
  • 计算机网络:应用层 —— 网络应用模式
  • @Repository注解和@mapper的区别
  • 解锁成长密码:探寻刻意练习之道
  • cuda-cuDnn
  • 如何使用Python和PIL库生成带竖排文字的封面图像
  • 低代码开发 实战转型案例一览
  • SQL Server中FIRST_VALUE和 LAST_VALUE窗口函数允许在一个指定的窗口内返回第一个或最后一个值