Android Telephony | SIM 卡管理和subId、slotId、phoneId 定义关系说明

本文主要是介绍Android Telephony | SIM 卡管理和subId、slotId、phoneId 定义关系说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

相关业务背景

DSDS双卡双待 :手机可以插入两张手机卡,而且能同时待机,但是某一时刻只能有一张卡上有电话。

一个SIM卡代表一个账号,那么双卡手机就意味着多账户,需要增加多账户管理和业务处理。

  • 账号管理:双卡手机提供界面给设置默认打电话、发短信和上网卡等。
  • 业务管理:基于两卡的独立性,双卡设备需要提供双卡进行通话业务设置(呼叫限制、呼叫转移)和网络业务设置(数据开关、数据漫游开关、VoLTE、VoWiFi业务开关等)。

和单卡业务类似,只是处理前需要选定一个账号,因此就需要正确的subId、slotId、phoneId等参数确认和区分业务卡。

SIM卡有关ID介绍

为了区分不同的卡,在Android上实现了多种id,如:slotId、phoneId、subId等,定义区别如下:

XID定义和特殊场景说明
ICCID

SIM 卡的唯一标识。

在SIM卡加载完成后,Telephony 以ICCID为关键字在数据库中查找,如果没有找到 ,则说明此卡第一次插入,则在数据库中添加此SIM卡记录(siminfo表);如果找到则更新原有记录,但是此记录在数据库中的序号不变,即subId不变。

Sub Id

siminfo表中的Primary Key,从1开始的整数,标识卡记录在数据库表中的顺序号。

第1张插入的卡是1,之后累加。

- SIM卡可以切换卡槽 ,但是subId不会变。phoneId和slotId会0/1变化。

- Android 平台,在手机关机时会将当前使用的SIM卡Slot ID设为-1,在下次开机时检测到加载成功的SIM卡后将对应的Slot ID设为0或1(这样当通过SubscriptionManager接口请求当前可用Sub info时,就能从db中检索SlotID不为-1的SIM卡即可)。

Phone IdPhone 实例对应的ID,双卡设备的phoneId是0或1。
Slot Id卡槽ID,当前可以认为slotId 和 phoneId 是相同的。

ID 变化规则和关系

ICCID、Sub ID 和 slot ID 都是 siminfo 数据库中的字段

  • Phone ID 和 Slot ID 当前可认为是一样的,跟卡槽对应,卡1的 ID 是0,卡2是1;
  • 切换卡槽,subId不变,phoneId 和 slotId 变化
  • SubscrptionManager 提供接口对 phoneId 进行转换。
    • subId 指示当前使用的SIM卡,phoneId/slotId 只关注卡槽。
API 函数名参数 ID隐藏/公开功能作用
getActiveSubscriptionInfo(int sudId)subId公开获取指定 subId 的当前可用的SIM卡信息
isNetworkRoaming(int subId)subId公开查询指定 subId 的漫游数据开关状态
getAcitiveSubscriptionInfoForSimSlotIndex(int slotIndex)slotId公开获取指定SlotId的当前可用SIM卡信息
getActiveSubscriptionInfoForIccIndex(String iccId)ICCIDhide获取指定ICCID的当前可用SIM卡信息
setDisplayName(String displayName, int subId)subIdhide将指定subId 的SIM卡显示名称保存到数据库
setDataRoaming(int roaming,  int subId)subIdhide将指定subId 的SIM卡漫游开关保存到数据库
getSlotIndex(int subId)subIdhide获取subId对应的slotId
getSubId(int slotIndex)subIdhide获取slotId对应的subId

数据库

双卡业务相关设置项分别保存在Settings 数据库、网络端(通话设置:呼叫转移和呼叫限制)和Telephony 数据库(telephony.db,需要权限)

获取telephony.db数据库文件:

adb  pull data/user_de/0/com.android.providers.telephony/databases

双卡设置在Settings 应用界面。

设置项子设置项Android KEY保存位置
双卡设置默认数据卡Settings.Global.MULTI_SIM_DATA_CALL_SUBSCRIPTIONSettings数据库
默认通话卡Settings.Global.MULTI_SIM_VOICE_CALL_SUBSCRIPTION
默认彩信卡Settings.Global.MULTI_SIM_SMS_SUBSCRIPTION
网络设置(卡 1和卡2)数据开关Settings.Global.Mobile_DATA + subId
数据漫游开关Settings.Global.DATA_ROAMING + subId
网络模式选择Settings.Global.PREFERRED_NETWORK_MODE + subId
VoLTE功能开关SubscriptiionManager.ENHANCED_4G_MODE_ENABLEDTelephony.db 的siminfo表
ViLTE功能开关SubscriptiionManager.VT_IMS_ENABLED
WFC功能开关

SubscriptiionManager.WFC_IMS_ENABLED

代码案例

在注册状态变化时(onSubscriptionsChanged)更新PhoneId。

    //添加注册状态变化的监听器private final SubscriptionManager.OnSubscriptionsChangedListener mOnSubscriptionsChangedListener =new SubscriptionManager.OnSubscriptionsChangedListener() {@Overridepublic void onSubscriptionsChanged() {//获取上下文信息final Activity activity = getActivity();if (activity == null || activity.isFinishing() || activity.isDestroyed()) {return;}final List<SubscriptionInfo> mSubList =SubscriptionUtil.getAvailableSubscriptions(getContext());SubscriptionInfo mSubInfo = null;for (SubscriptionInfo subInfo : mSubList) {Log.d(TAG, "onSubscriptionsChanged: subInfo = " + subInfo.toString());if (subInfo.getSubscriptionId() == mSubId) {mSubInfo = subInfo;//校验PhoneId正常并更新的方法if (!SubscriptionManager.isValidPhoneId(mPhoneId)) {mPhoneId = SubscriptionManager.getPhoneId(mSubId);Log.d(TAG, "onSubscriptionsChanged: mPhoneId = " + mPhoneId);if (SubscriptionManager.isValidPhoneId(mPhoneId)) {mPhone = PhoneFactory.getPhone(mPhoneId);}}break;}}if (mSubInfo == null) {Log.d(TAG, "onSubscriptionsChanged: can't find subInfo for subId " + mSubId, so call finish()");}}};

通过subId获取phoneId。疑问:通过subid获取phoneId为什么会异常?
 

        if (mSubId != SubscriptionManager.INVALID_SUBSCRIPTION_ID) {int mPhoneId = SubscriptionManager.getPhoneId(mSubId); //会异常啊if (!SubscriptionManager.isValidPhoneId(mPhoneId)) {mPhoneId = getPhoneIdFromSubId(mSubId, phoneId);log("getPhone: phoneIdFromSubId: " + phoneId);}//phoneId合法,则根据ID获取Phone对象if (SubscriptionManager.isValidPhoneId(phoneId)) {log("updatePhone :: isValidPhoneId: ");mPhone = PhoneFactory.getPhone(phoneId);}}//类似源码命名:getActiveSubscriptionInfoForIccIndex,通过iccid获取subId,用for。//通过subId 获取PhoneId(源码命名是For的)private int getPhoneIdFromSubId(int subId, int phoneId){int phoneIdFromSubId = phoneId;//获取注册信息列表List<SubscriptionInfo> subscriptionInfoList = mSubscriptionManager.getAvailableSubscriptionInfoList();String iccidForSub = null;if (subscriptionInfoList != null) {for (SubscriptionInfo si : subscriptionInfoList) {if (si.getSubscriptionId() == subId){iccidForSub = si.getIccId();break;}}}List<UiccCardInfo> cardsInfos = mTelephonyManager.getUiccCardsInfo();if (cardsInfo != null) {for (UiccCardInfo info : cardsInfos) {if (info != null) {String iccid = info.getIccId();if (iccidForSub != null && iccidForSub.equals(iccid)) {phoneIdFromSubId = info.getSlotIndex();break;}}}}return phoneIdFromSubId;}

这篇关于【笔记】Android Telephony | SIM 卡管理和subId、slotId、phoneId 定义关系说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!

这篇关于Android Telephony | SIM 卡管理和subId、slotId、phoneId 定义关系说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/828164

相关文章

Zookeeper安装和配置说明

一、Zookeeper的搭建方式 Zookeeper安装方式有三种,单机模式和集群模式以及伪集群模式。 ■ 单机模式:Zookeeper只运行在一台服务器上,适合测试环境; ■ 伪集群模式:就是在一台物理机上运行多个Zookeeper 实例; ■ 集群模式:Zookeeper运行于一个集群上,适合生产环境,这个计算机集群被称为一个“集合体”(ensemble) Zookeeper通过复制来实现

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

git使用的说明总结

Git使用说明 下载安装(下载地址) macOS: Git - Downloading macOS Windows: Git - Downloading Windows Linux/Unix: Git (git-scm.com) 创建新仓库 本地创建新仓库:创建新文件夹,进入文件夹目录,执行指令 git init ,用以创建新的git 克隆仓库 执行指令用以创建一个本地仓库的

POJ1269 判断2条直线的位置关系

题目大意:给两个点能够确定一条直线,题目给出两条直线(由4个点确定),要求判断出这两条直线的关系:平行,同线,相交。如果相交还要求出交点坐标。 解题思路: 先判断两条直线p1p2, q1q2是否共线, 如果不是,再判断 直线 是否平行, 如果还不是, 则两直线相交。  判断共线:  p1p2q1 共线 且 p1p2q2 共线 ,共线用叉乘为 0  来判断,  判断 平行:  p1p

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

log4j2相关配置说明以及${sys:catalina.home}应用

${sys:catalina.home} 等价于 System.getProperty("catalina.home") 就是Tomcat的根目录:  C:\apache-tomcat-7.0.77 <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss} [%t] %-5p %c{1}:%L - %msg%n" /> 2017-08-10