android公开的API提供了访问方法,大家都知道使用TelephonyManager提供的方法,但是有些理解有误,如下国内一个比较大的andorid论坛提供的例子,就出现了错误:
帖子如下http://www.eoeandroid.com/thread-14027-1-3.html,其中实现代码没有注释,只能按照变量定义判断:
TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY_SERVICE);
String deviceid = tm.getDeviceId();
String tel =tm.getLine1Number();//取出用户手机号码?
String imei =tm.getSimSerialNumber();//取出IMEI?
String imsi=tm.getSubscriberId();//取出IMSI?
那么上述出现错误了:String imei =tm.getSimSerialNumber();//取出IMEI
IMEI是手机的序列号,怎么会通过getSimSerialNumber()方法获得,那么查一下andorid源码可以看出:
http://www.netmite.com/android/mydroid/frameworks/base/telephony/java/android/telephony/TelephonyManager.java
从注释里明显看出来这个函数是取SIM卡序列号的,也就是ICCID的,他用错了。
3.那好我们看看andorid实现TelephonyManager.java的源码:
public String getSimSerialNumber() {
try {
return getSubscriberInfo().getSimSerialNumber();
} catch (RemoteException ex) {
}
return null;
}
2.相关几个定义、说明:
我们说到的和手机、卡相关的号码数据包括IMSI,MSISDN,ICCID,IMEI
IMSI:international mobiles subscriber identity国际移动用户号码标识,
这个一般大家是不知道,GSM必须写在卡内相关文件中;
MSISDN:mobile subscriber ISDN用户号码,这个是我们说的139,136那个号码;
ICCID:ICC identity集成电路卡标识,这个是唯一标识一张卡片物理号码的;
IMEI:international mobile Equipment identity手机唯一标识码;
getDeviceId()取IMEI号没有争议了。
public String getDeviceId(){
try {
returngetSubscriberInfo().getDeviceId();
} catch(RemoteException ex) {
}
returnnull;
}
getLine1Number()取MSISDN,这个需要说明两点,1为什么这个函数叫getLine1Number(),因为andorid实现的时候应该分为GSM和CDMA的,GSM手机使用这个函数,CDMA应该还会由其它实现的。
2取MSISDN具体的方法就会导致最后能否取到了,函数中调用了getSubscriberInfo().getLine1Number()去实现,我们下面找找看。
public StringgetLine1Number() {
try {
returngetSubscriberInfo().getLine1Number();
} catch(RemoteException ex) {
}
returnnull;
}
找到了private IPhoneSubInfo getSubscriberInfo() {
// get iteach time because that process crashes a lot
returnIPhoneSubInfo.Stub.asInterface(ServiceManager.getService("iphonesubinfo"));
}
一个接口,再找有一个PhoneSubInfo.java:
public String getDeviceId(){
mContext.enforceCallingOrSelfPermission(READ_PHONE_STATE,"Requires READ_PHONE_STATE");
returnmPhone.getDeviceId();
}
前面定义了Phone mPhone,再找Phone.java:
StringgetSubscriberId();
原来是个接口,发现PhoneProxy.java有具体实现
public String getSubscriberId() {
returnmActivePhone.getSubscriberId();
}
这个mActivePhone是phone的实例,我疯了,于是发现GSMPHONE。java中有了具体实现:
public StringgetSubscriberId() {
returnmSIMRecords.imsi;
}
public StringgetIccSerialNumber() {
returnmSIMRecords.iccid;
}
public StringgetLine1Number() {
returnmSIMRecords.getMsisdnNumber();
从上面看出来,应该是通过SIM卡相关文件记录得到的上述数据,从其中看到:
public void handleMessage(Message msg) 这个函数进行了真正的处理,重点看:
case EVENT_GET_MSISDN_DONE:
isRecordLoadResponse = true;
ar =(AsyncResult)msg.obj;
if(ar.exception != null) {
Log.d(LOG_TAG,"Invalid or missing EF[MSISDN]");//应该是从sim卡的EFmsisdn文件中取出来的
break;
}
adn =(AdnRecord)ar.result;
msisdn =adn.getNumber();
msisdnTag =adn.getAlphaTag();
Log.d(LOG_TAG, "MSISDN: " + msisdn);
break;
下面的细节就不分析了,那个问题就归结到是否可以从SIM卡的EFmsisdn文件取出手机号码了,不幸的是一般运营商不会把用户号码写在这个文件的,为什么呢?
因为这个手机号码是在用户买到卡并开通时才将IMSI和MSISDN对应上的,卡内生产出来时只有IMSI,你不知道用户喜欢那个手机号码,因此一般不先对应IMSI和MSISDN,即时有对应也不写这个文件的。
4.总结一下:
TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY_SERVICE);
String imei =tm.getDeviceId();//取出IMEI
String tel =tm.getLine1Number();//取出MSISDN,很可能为空
String imei =tm.getSimSerialNumber();//取出ICCID
String imsi=tm.getSubscriberId();//取出IMSI