【笔记】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 Id Phone 实例对应的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 只关注卡槽。
  • 每张SIM卡都有一个对应的Subscript
  • 20
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要监听 Android 设备的 SIM 卡状态,可以使用 `PhoneStateListener` 类。以下是一个示例代码: ```java import android.content.Context; import android.telephony.PhoneStateListener; import android.telephony.TelephonyManager; import android.util.Log; public class SimStateReceiver { private static final String TAG = "SimStateReceiver"; private Context mContext; private TelephonyManager mTelephonyManager; private PhoneStateListener mPhoneStateListener; public SimStateReceiver(Context context) { mContext = context; mTelephonyManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE); mPhoneStateListener = new PhoneStateListener() { @Override public void onCallStateChanged(int state, String phoneNumber) { super.onCallStateChanged(state, phoneNumber); Log.d(TAG, "onCallStateChanged: state = " + state + ", phoneNumber = " + phoneNumber); } @Override public void onServiceStateChanged(ServiceState serviceState) { super.onServiceStateChanged(serviceState); Log.d(TAG, "onServiceStateChanged: serviceState = " + serviceState); } @Override public void onSimStateChanged(int state) { super.onSimStateChanged(state); Log.d(TAG, "onSimStateChanged: state = " + state); switch (state) { case TelephonyManager.SIM_STATE_ABSENT: // SIM 卡不存在或未插入 break; case TelephonyManager.SIM_STATE_NETWORK_LOCKED: // SIM 卡已被网络锁定 break; case TelephonyManager.SIM_STATE_PIN_REQUIRED: // SIM 卡需要 PIN 解锁 break; case TelephonyManager.SIM_STATE_PUK_REQUIRED: // SIM 卡需要 PUK 解锁 break; case TelephonyManager.SIM_STATE_READY: // SIM 卡已准备好 break; case TelephonyManager.SIM_STATE_UNKNOWN: // SIM 卡状态未知 break; } } }; } public void start() { mTelephonyManager.listen(mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE | PhoneStateListener.LISTEN_SERVICE_STATE | PhoneStateListener.LISTEN_SIM_STATE); } public void stop() { mTelephonyManager.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE); } } ``` 在上面的代码中,我们创建了一个 `SimStateReceiver` 类,它有一个 `start()` 方法和一个 `stop()` 方法,用于开始和停止监听 SIM 卡状态。在 `SimStateReceiver` 的构造方法中,我们创建了一个 `PhoneStateListener` 对象,并重写了其中的 `onSimStateChanged()` 方法,用于监听 SIM 卡状态的变化。在 `onSimStateChanged()` 方法中,我们根据不同的 SIM 卡状态打印不同的日志信息。最后,在 `start()` 方法中,我们通过 `TelephonyManager` 的 `listen()` 方法注册了 `PhoneStateListener` 对象,开始监听 SIM 卡状态。在 `stop()` 方法中,我们通过 `TelephonyManager` 的 `listen()` 方法取消了 `PhoneStateListener` 对象的注册,停止监听 SIM 卡状态。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值