2023年12月11日发(作者:赫飇)
Android获取双卡手机IMEI,IMSI,ICCID
一、首先要添加权限
二、获取主卡的IMEI,IMSI,ICCID
/**
* Author: liuqiang
* Time: 2017-08-14 15:28
* Description:
*
* IMEI 与你的手机是绑定关系 用于区别移动终端设备
* IMSI 与你的手机卡是绑定关系 用于区别移动用户的有效信息 IMSI是用户的标识。
* ICCID ICCID是卡的标识,由20位数字组成
* ICCID只是用来区别SIM卡,不作接入网络的鉴权认证。而IMSI在接入网络的时候,会到运营商的服务器中进行验证。
* /android/platform_frameworks_base/blob/master/telephony/java/android/telephony/
*/
@RequiresApi(api = N_CODES.O)
public void check(View view) {
TelephonyManager telephonyManager = (TelephonyManager) this
.getSystemService(TELEPHONY_SERVICE);// 取得相关系统服务
String simOperatorName = OperatorName();
String imei = iceId(); //取出 IMEI
String imeiAPI26 = i(); //取出 IMEI 需要 api26以上
String tel = e1Number(); //取出 MSISDN,很可能为空
String imsi = scriberId(); //取出 IMSI
String icc = SerialNumber(); //取出 ICCID
Log.d("Q_M", "运行商名字--" + simOperatorName);
Log.d("Q_M", "IMEI--" + imei);
Log.d("Q_M", "IMEI_API26--" + imeiAPI26);
Log.d("Q_M", "IMSI--" + imsi);
Log.d("Q_M", "ICCID--" + icc);
}
三、如果手机有多张卡
其实多卡情况下主要要获得的是两个地方:
getSubscriberId
和
getSimSerialNumber
,打开上面的的源码,搜索一下这两个方法,发现这两个方法都有
一个带参数(subId)的重载方法,并且这两个方法都是
@hide
的,hide倒是无所谓,这个可以通过反射调用,主要的问题要弄清楚他的这个参数
subId是个什么东西。代码片段如下:
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*/
public String getSubscriberId() {
return getSubscriberId(getSubId());
}
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone
* for a subscription.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*
* @param subId whose subscriber id is returned
* @hide
*/
public String getSubscriberId(int subId) {
try {
IPhoneSubInfo info = getSubscriberInfo();
if (info == null)
return null;
return scriberIdForSubscriber(subId, ackageName());
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
0 、不算业余知识的业余知识
既然所有的CURD都是针对这个数据库,那么来研究下这个数据库,
打开后发现这个数据库中有张叫
siminfo
的表。如下图:
截图中并没有包含表中所有的列,但是这些是主要的,后面的那些和这篇文章无关。上图这个是在Android模拟器中导出来的。在看一张表的时
候,不仅要了解他的字段的含义,而且要了解这种表中数据的插入规则。
很重要:每次插入一张新SIM卡(这张卡没有插入到过这个手机)的时候,都会在数据库中添加一条记录,插入的时
候
_id
字段从1开始每次增1;每次移除一张SIM卡,
sim_id
字段设置为
-1
(但是这条记录不会被删除),表示这张卡曾经
插入过,但是又被移除了,用专业术语来说就是这张卡现在不是
Active
的,处于不可用状态。
这些字段都表示了什么意思,其中最重要的是
_id
和
sim_id
:
_id:从数据库的角度来说,做过sqlite开发的都知道,他是个从1开始自增的主键。但是他在这里还代表了程序中另一个东西
subId
也就
是
subscription id
icc_id:不解释,上面说过了
sim_id:这个字段有两层含义,在大于-1,的情况下他表示的是卡槽序号,比如sim_id为0表示
卡1
,取值为1的时候表示的是
卡2
,以此类推,
但是一般手机不会超过两个卡槽吧?!如果取值为-1,表示这张SIM卡曾经被插入过,但是现在被移除了。
display_name:顾名思义,显示名。这个一般可以改,但是默认的是读取的运营商的名字,比如:中国移动,中国联通,中国电信
carrier_name :恩,运营商名字
number:SIM卡对应的手机号,这个不一定能取到
mcc:Mobile Country Code,移动国家码
mnc:Mobile Network Code,移动网络码
现在的主要问题是如何获取
subscription id
,对吧,how?
1 、读数据库取subId也即是表的_id 字段
///data/data/ony/databases/
public void getSimInfo() {
Uri uri = ("content://telephony/siminfo");
Cursor cursor = null;
ContentResolver contentResolver = getApplicationContext().getContentResolver();
cursor = (uri,
new String[]{"_id", "sim_id", "icc_id", "display_name"}, "0=0",
new String[]{}, null);
if (null != cursor) {
while (Next()) {
String icc_id = ing(umnIndex("icc_id"));
String display_name = ing(umnIndex("display_name"));
int sim_id = (umnIndex("sim_id"));
int _id = (umnIndex("_id"));
Log.d("Q_M", "icc_id-->" + icc_id);
Log.d("Q_M", "sim_id-->" + sim_id);
Log.d("Q_M", "display_name-->" + display_name);
Log.d("Q_M", "subId或者说是_id->" + _id);
Log.d("Q_M", "---------------------------------");
}
}
}
如上代码,我在小米6的测试机上进行测试,插入过3张卡,两张移动,一张联通的,运行结果如下(因为是真是的SIM卡,隐藏了icc_id):
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898600*************7
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->0
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->中国移动
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->1
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898601*************6
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->-1
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->CARD 2
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->2
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898602*************9
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->1
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->中国移动
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->3
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
2、通过SubscriptionManager取subId
但是这个api有两个缺陷:必须5.0以上才能用;只能获取active的卡的信息。其实对于第二点缺陷,源码中是能获取所有的卡的信息的,也就是表
中所有的数据,通过源码看到这个方法叫做
getAllSubscriptionInfoList
不过这个方法也被hide了,能用是能用,就是这种@hide的方法,不一定在哪
个版本就会被删除掉。需要做很多兼容性上的操作。
@RequiresApi(api = N_OP_MR1)
public void getSimInfoBySubscriptionManager() {
List
for (SubscriptionInfo info : list) {
Log.d("Q_M", "ICCID-->" + Id());
Log.d("Q_M", "subId-->" + scriptionId());
Log.d("Q_M", "DisplayName-->" + playName());
Log.d("Q_M", "CarrierName-->" + rierName());
Log.d("Q_M", "---------------------------------");
}
}
其实代码分析到这里,双卡的其他基本信息其实都已经获取到了,无论通过读取数据库的方式,还是通过
SubscriptionManager
的方式,唯一没有获
取到的就是
getSubscriberId()
来得到的IMSI。
3、通过反射获取调用IMSI
反射调用带有参数的
getSubscriberId(subId)
是很简单的,但是问题就出在,这个方法是@hide,所以说这个方法可能在不同的Android版本中会出
现不同的实现。subId在5.0传入的是long类型的参数,而5.1-7.1.1传入的是int类型的参数。再高的版本我就没看了
5.0.0上的方法签名:
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone
* for a subscription.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*
* @param subId whose subscriber id is returned
*/
/** {@hide} */
public String getSubscriberId(long subId) {
try {
return getSubscriberInfo().getSubscriberIdForSubscriber(subId);
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
5.1.0-5.1.1的方法签名:
public String getSubscriberId(int subId) {
try {
return getSubscriberInfo().getSubscriberIdForSubscriber(subId);
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
6.0.0-7.1.1的方法签名:
public String getSubscriberId(int subId) {
try {
IPhoneSubInfo info = getSubscriberInfo();
if (info == null)
return null;
return scriberIdForSubscriber(subId, ackageName());
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
反射调用
getSubscriberId
的代码如下,其中参数,subId的获取方式,在上面已经有了:
public String getSubscriberId(int subId) {
TelephonyManager telephonyManager = (TelephonyManager) this
.getSystemService(TELEPHONY_SERVICE);// 取得相关系统服务
Class> telephonyManagerClass = null;
String imsi = null;
try {
telephonyManagerClass = e("onyManager");
if (_INT > N_OP) {
Method method = hod("getSubscriberId", );
imsi = (String) (telephonyManager, subId);
} else if (_INT == N_OP) {
Method method = hod("getSubscriberId", );
imsi = (String) (telephonyManager, (long) subId);
}
} catch (Exception e) {
tackTrace();
}
Log.d("Q_M", "IMSI--" + imsi);
return imsi;
}
最后: 这些操作都是在Android5.0以上版本的,并且第三方厂商没有修改这块的代码的情况下的操作。至于5.0以下的
系统,怎么判断,好像要分平台单个处理
Android4.x上处理双卡
高通平台有单独的类
lephonyManager
里面有方法,这个可以用反射的方式调用,经过验证在努比亚4.4.2的系统上是可以
获取的:
getSubscriberId(int simId)
getSimSerialNumber(int simId)
测试代码如下:
/**
* Author: liuqiang
* Time: 2017-08-15 10:56
* Description:
*
* 高通的“lephonyManager”类
*/
private void getAPI19SimInfo() {
Class> tm = null;
try {
tm = e("lephonyManager");
Method getSubscriberIdMethod = hod("getSubscriberId", );
Method getSimSerialNumberMethod = hod("getSimSerialNumber", );
Object service = temService("phone_msim");
//0 代表卡1
//1 代表卡2
String s = (String) (service, 0);
Log.d("Q_M", "-->" + s);
} catch (ClassNotFoundException e) {
tackTrace();
} catch (NoSuchMethodException e) {
tackTrace();
} catch (InvocationTargetException e) {
tackTrace();
} catch (IllegalAccessException e) {
tackTrace();
}
}
至于MTK平台,据说有个aidl""类或者用Manager
类
onyManagerEx
,我这没有MTK平台的测试机,没法测。
【参考文章】
/tencent_bugly/article/details/51911047
/flyme/p/
/peijiangping1989/article/details/20448007
2023年12月11日发(作者:赫飇)
Android获取双卡手机IMEI,IMSI,ICCID
一、首先要添加权限
二、获取主卡的IMEI,IMSI,ICCID
/**
* Author: liuqiang
* Time: 2017-08-14 15:28
* Description:
*
* IMEI 与你的手机是绑定关系 用于区别移动终端设备
* IMSI 与你的手机卡是绑定关系 用于区别移动用户的有效信息 IMSI是用户的标识。
* ICCID ICCID是卡的标识,由20位数字组成
* ICCID只是用来区别SIM卡,不作接入网络的鉴权认证。而IMSI在接入网络的时候,会到运营商的服务器中进行验证。
* /android/platform_frameworks_base/blob/master/telephony/java/android/telephony/
*/
@RequiresApi(api = N_CODES.O)
public void check(View view) {
TelephonyManager telephonyManager = (TelephonyManager) this
.getSystemService(TELEPHONY_SERVICE);// 取得相关系统服务
String simOperatorName = OperatorName();
String imei = iceId(); //取出 IMEI
String imeiAPI26 = i(); //取出 IMEI 需要 api26以上
String tel = e1Number(); //取出 MSISDN,很可能为空
String imsi = scriberId(); //取出 IMSI
String icc = SerialNumber(); //取出 ICCID
Log.d("Q_M", "运行商名字--" + simOperatorName);
Log.d("Q_M", "IMEI--" + imei);
Log.d("Q_M", "IMEI_API26--" + imeiAPI26);
Log.d("Q_M", "IMSI--" + imsi);
Log.d("Q_M", "ICCID--" + icc);
}
三、如果手机有多张卡
其实多卡情况下主要要获得的是两个地方:
getSubscriberId
和
getSimSerialNumber
,打开上面的的源码,搜索一下这两个方法,发现这两个方法都有
一个带参数(subId)的重载方法,并且这两个方法都是
@hide
的,hide倒是无所谓,这个可以通过反射调用,主要的问题要弄清楚他的这个参数
subId是个什么东西。代码片段如下:
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*/
public String getSubscriberId() {
return getSubscriberId(getSubId());
}
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone
* for a subscription.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*
* @param subId whose subscriber id is returned
* @hide
*/
public String getSubscriberId(int subId) {
try {
IPhoneSubInfo info = getSubscriberInfo();
if (info == null)
return null;
return scriberIdForSubscriber(subId, ackageName());
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
0 、不算业余知识的业余知识
既然所有的CURD都是针对这个数据库,那么来研究下这个数据库,
打开后发现这个数据库中有张叫
siminfo
的表。如下图:
截图中并没有包含表中所有的列,但是这些是主要的,后面的那些和这篇文章无关。上图这个是在Android模拟器中导出来的。在看一张表的时
候,不仅要了解他的字段的含义,而且要了解这种表中数据的插入规则。
很重要:每次插入一张新SIM卡(这张卡没有插入到过这个手机)的时候,都会在数据库中添加一条记录,插入的时
候
_id
字段从1开始每次增1;每次移除一张SIM卡,
sim_id
字段设置为
-1
(但是这条记录不会被删除),表示这张卡曾经
插入过,但是又被移除了,用专业术语来说就是这张卡现在不是
Active
的,处于不可用状态。
这些字段都表示了什么意思,其中最重要的是
_id
和
sim_id
:
_id:从数据库的角度来说,做过sqlite开发的都知道,他是个从1开始自增的主键。但是他在这里还代表了程序中另一个东西
subId
也就
是
subscription id
icc_id:不解释,上面说过了
sim_id:这个字段有两层含义,在大于-1,的情况下他表示的是卡槽序号,比如sim_id为0表示
卡1
,取值为1的时候表示的是
卡2
,以此类推,
但是一般手机不会超过两个卡槽吧?!如果取值为-1,表示这张SIM卡曾经被插入过,但是现在被移除了。
display_name:顾名思义,显示名。这个一般可以改,但是默认的是读取的运营商的名字,比如:中国移动,中国联通,中国电信
carrier_name :恩,运营商名字
number:SIM卡对应的手机号,这个不一定能取到
mcc:Mobile Country Code,移动国家码
mnc:Mobile Network Code,移动网络码
现在的主要问题是如何获取
subscription id
,对吧,how?
1 、读数据库取subId也即是表的_id 字段
///data/data/ony/databases/
public void getSimInfo() {
Uri uri = ("content://telephony/siminfo");
Cursor cursor = null;
ContentResolver contentResolver = getApplicationContext().getContentResolver();
cursor = (uri,
new String[]{"_id", "sim_id", "icc_id", "display_name"}, "0=0",
new String[]{}, null);
if (null != cursor) {
while (Next()) {
String icc_id = ing(umnIndex("icc_id"));
String display_name = ing(umnIndex("display_name"));
int sim_id = (umnIndex("sim_id"));
int _id = (umnIndex("_id"));
Log.d("Q_M", "icc_id-->" + icc_id);
Log.d("Q_M", "sim_id-->" + sim_id);
Log.d("Q_M", "display_name-->" + display_name);
Log.d("Q_M", "subId或者说是_id->" + _id);
Log.d("Q_M", "---------------------------------");
}
}
}
如上代码,我在小米6的测试机上进行测试,插入过3张卡,两张移动,一张联通的,运行结果如下(因为是真是的SIM卡,隐藏了icc_id):
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898600*************7
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->0
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->中国移动
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->1
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898601*************6
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->-1
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->CARD 2
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->2
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
08-14 16:46:11.208 11583-11583/eck D/Q_M: icc_id-->898602*************9
08-14 16:46:11.208 11583-11583/eck D/Q_M: sim_id-->1
08-14 16:46:11.208 11583-11583/eck D/Q_M: display_name-->中国移动
08-14 16:46:11.208 11583-11583/eck D/Q_M: subId或者说是_id->3
08-14 16:46:11.208 11583-11583/eck D/Q_M: ---------------------------------
2、通过SubscriptionManager取subId
但是这个api有两个缺陷:必须5.0以上才能用;只能获取active的卡的信息。其实对于第二点缺陷,源码中是能获取所有的卡的信息的,也就是表
中所有的数据,通过源码看到这个方法叫做
getAllSubscriptionInfoList
不过这个方法也被hide了,能用是能用,就是这种@hide的方法,不一定在哪
个版本就会被删除掉。需要做很多兼容性上的操作。
@RequiresApi(api = N_OP_MR1)
public void getSimInfoBySubscriptionManager() {
List
for (SubscriptionInfo info : list) {
Log.d("Q_M", "ICCID-->" + Id());
Log.d("Q_M", "subId-->" + scriptionId());
Log.d("Q_M", "DisplayName-->" + playName());
Log.d("Q_M", "CarrierName-->" + rierName());
Log.d("Q_M", "---------------------------------");
}
}
其实代码分析到这里,双卡的其他基本信息其实都已经获取到了,无论通过读取数据库的方式,还是通过
SubscriptionManager
的方式,唯一没有获
取到的就是
getSubscriberId()
来得到的IMSI。
3、通过反射获取调用IMSI
反射调用带有参数的
getSubscriberId(subId)
是很简单的,但是问题就出在,这个方法是@hide,所以说这个方法可能在不同的Android版本中会出
现不同的实现。subId在5.0传入的是long类型的参数,而5.1-7.1.1传入的是int类型的参数。再高的版本我就没看了
5.0.0上的方法签名:
/**
* Returns the unique subscriber ID, for example, the IMSI for a GSM phone
* for a subscription.
* Return null if it is unavailable.
*
* Requires Permission:
* {@link sion#READ_PHONE_STATE READ_PHONE_STATE}
*
* @param subId whose subscriber id is returned
*/
/** {@hide} */
public String getSubscriberId(long subId) {
try {
return getSubscriberInfo().getSubscriberIdForSubscriber(subId);
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
5.1.0-5.1.1的方法签名:
public String getSubscriberId(int subId) {
try {
return getSubscriberInfo().getSubscriberIdForSubscriber(subId);
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
6.0.0-7.1.1的方法签名:
public String getSubscriberId(int subId) {
try {
IPhoneSubInfo info = getSubscriberInfo();
if (info == null)
return null;
return scriberIdForSubscriber(subId, ackageName());
} catch (RemoteException ex) {
return null;
} catch (NullPointerException ex) {
// This could happen before phone restarts due to crashing
return null;
}
}
反射调用
getSubscriberId
的代码如下,其中参数,subId的获取方式,在上面已经有了:
public String getSubscriberId(int subId) {
TelephonyManager telephonyManager = (TelephonyManager) this
.getSystemService(TELEPHONY_SERVICE);// 取得相关系统服务
Class> telephonyManagerClass = null;
String imsi = null;
try {
telephonyManagerClass = e("onyManager");
if (_INT > N_OP) {
Method method = hod("getSubscriberId", );
imsi = (String) (telephonyManager, subId);
} else if (_INT == N_OP) {
Method method = hod("getSubscriberId", );
imsi = (String) (telephonyManager, (long) subId);
}
} catch (Exception e) {
tackTrace();
}
Log.d("Q_M", "IMSI--" + imsi);
return imsi;
}
最后: 这些操作都是在Android5.0以上版本的,并且第三方厂商没有修改这块的代码的情况下的操作。至于5.0以下的
系统,怎么判断,好像要分平台单个处理
Android4.x上处理双卡
高通平台有单独的类
lephonyManager
里面有方法,这个可以用反射的方式调用,经过验证在努比亚4.4.2的系统上是可以
获取的:
getSubscriberId(int simId)
getSimSerialNumber(int simId)
测试代码如下:
/**
* Author: liuqiang
* Time: 2017-08-15 10:56
* Description:
*
* 高通的“lephonyManager”类
*/
private void getAPI19SimInfo() {
Class> tm = null;
try {
tm = e("lephonyManager");
Method getSubscriberIdMethod = hod("getSubscriberId", );
Method getSimSerialNumberMethod = hod("getSimSerialNumber", );
Object service = temService("phone_msim");
//0 代表卡1
//1 代表卡2
String s = (String) (service, 0);
Log.d("Q_M", "-->" + s);
} catch (ClassNotFoundException e) {
tackTrace();
} catch (NoSuchMethodException e) {
tackTrace();
} catch (InvocationTargetException e) {
tackTrace();
} catch (IllegalAccessException e) {
tackTrace();
}
}
至于MTK平台,据说有个aidl""类或者用Manager
类
onyManagerEx
,我这没有MTK平台的测试机,没法测。
【参考文章】
/tencent_bugly/article/details/51911047
/flyme/p/
/peijiangping1989/article/details/20448007