安卓源码避坑指南5——获取通话记录或通讯录的数量为负值

安卓源码避坑指南5——获取通话记录或通讯录的数量为负值

在这里插入图片描述

通过蓝牙PBAP协议同步通讯录、通话记录时,想必对其数量大小也是很感兴趣的,因此一般的设计思路都是先获取到同步对象的总大小,然后再同步该对象的具体数据。

想法是美好的,然而现实却很魔幻。获取同步对象的总数量这块最近就遇到个奇怪问题——获取到的总数量为负值,今天就和大家一探究竟,详解此类问题。

测试环境:android-9车机系统(高通安卓源码)

测试步骤

  1. 车机蓝牙配对连接手机
  2. 车机同步通讯录、通话记录等(手机授权通讯录权限)

测试现象:获取到的通话记录数量为负值,由于应用内部逻辑,错误的负值不再同步该对象,通话记录为空。

What?怎么会获取到一个负值呢?带着疑问开始接下来的分析。

首先根据HCI层的交互,可以确定手机回复的数据是正确的,也就是说手机给的通话记录的size是个正数:
在这里插入图片描述

当我看到这个数量时,我心里萌生出这个想法:我艹,这哥们这么能打电话嘛…

而蓝牙服务层在接收到OBEX的回复消息后解析成了负值:
在这里插入图片描述

根据OBEX协议定义,同步对象的size大小使用两字节表示,所以 41708 的二进制表示法为:0b1010 0010 1110 1100,而 -23828 的二进制表示法也为:0b1010 0010 1110 1100

对正负数在计算机中存储形式还不清楚的同学可以打电话给自己的大学计算机老师了(哈哈),这样就可以解释通话记录变为负值的原因了。java中的数据是有符号区分的,两个字节的short类型数据的范围为:-2^15 ~ 2^15-1。而 41708 这个数值很明显已经超出 short类型值的范围,因此溢出被解析成负值。

安卓源码中解析该数据的方法为:com.android.bluetooth.pbapclient.ObexAppParameters. getShort(),当byte型数组中存储的两字节数据超出short类型值的范围就会解析成异常值。
在这里插入图片描述

解决方案:获取到负值肯定不是我们所希望的,那如何才能正确获取某个同步对象的size大小值呢?java中没有提供相关方法来获取无符号的数值,但是我们可以通过数值操作间接获取到该数据对应的无符号值。

以上方法获取到的short类型值按位与上0xFFFF后赋值给到一个int类型的变量,则该int类型变量表示的值则为正确的short类型无符号值:
在这里插入图片描述

问题延伸
此类问题就是没有考虑到数值类型中有符号和无符号数据在值范围上的差异,同样地获取通讯录数量大小也是存在该风险的。最后的总结:数值由无符号转化为有符号时需注意数值是否有溢出风险

感兴趣的小伙伴欢迎私信留言一起讨论,源码避坑,永无止境,共同学习,一起进步!

更多互联互通技术,欢迎关注微信公众号:Connectivity
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ActivityManager是Android系统中非常重要的一个组件,负责管理系统中所有Activity的生命周期、进程管理、任务管理等。在本篇文章中,我们将深入了解ActivityManager框架的实现原理。 ## Activity的启动过程 在Android系统中,Activity是应用程序中最基本的界面单元。当用户点击应用程序的图标时,ActivityManager会负责启动应用程序并启动相应的Activity。下面是Activity的启动过程: 1. 应用程序进程启动:ActivityManager会创建一个新的进程来运行应用程序。 2. 应用程序的Application对象创建:在应用程序进程启动之后,ActivityManager会负责创建应用程序的Application对象。 3. Activity对象创建:当用户点击应用程序的图标并选择启动Activity时,ActivityManager会负责创建相应的Activity对象。 4. Activity的onCreate方法调用:ActivityManager会调用Activity对象的onCreate方法,完成Activity的初始化工作。 5. Activity的onStart方法调用:ActivityManager会调用Activity对象的onStart方法,将Activity显示给用户。 6. Activity的onResume方法调用:ActivityManager会调用Activity对象的onResume方法,让Activity成为用户交互的焦点。 7. Activity的生命周期结束:当用户退出Activity时,ActivityManager会负责销毁Activity对象。 ## 进程管理 在Android系统中,每个应用程序都运行在一个独立的进程中。ActivityManager负责管理所有进程,以确保系统的稳定和性能。 当应用程序被启动时,ActivityManager会为其创建一个新的进程并分配一定的内存资源。如果此时系统内存不足,ActivityManager会根据一定的策略来回收一些进程内存,以确保系统正常运行。 ActivityManager还负责监控进程的CPU使用情况和内存使用情况。如果一个进程的CPU使用率过高或者内存使用量过大,ActivityManager会考虑回收该进程的资源,以避免系统崩溃。 ## 任务管理 在Android系统中,任务是指一组相关的Activity。当用户启动一个应用程序时,ActivityManager会创建一个新的任务,并将应用程序的第一个Activity加入到该任务中。 当用户从一个Activity切换到另一个Activity时,ActivityManager会将前一个Activity加入到任务的后台,并将新的Activity显示给用户。当用户点击返回键时,ActivityManager会将当前Activity从任务中移除,并将前一个Activity重新显示给用户。 需要注意的是,当应用程序中的所有Activity都被销毁时,该应用程序的任务也会被销毁。这是因为Android系统中的Activity都是基于任务的,一个应用程序的所有Activity都属于同一个任务。 ## 总结 ActivityManager是Android系统中非常重要的一个组件,它负责管理系统中所有Activity的生命周期、进程管理、任务管理等。在本篇文章中,我们深入了解了ActivityManager的实现原理。熟悉ActivityManager的工作原理对于开发高质量的Android应用程序非常重要。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值