前言
短信中心号码概念
短信中心号码类似信息服务器,如果信息中心号码不正确,短消息是无法发送成功的,各个地区都有自己的信息中心号码,其中移动以+86138开头,关于该号码的获取这举例说明。
获取方法
- 1、调用Phone中的getSmscAddress(Message message)方法,其中参数的message为获取到结果后发送消息给mHandler,并查询结果AsyncResult的result属性中。
phone=PhoneFactory.getDefaultPhone(); phone.getSmscAddress(mHandler.obtainMessage(EVENT_QUERY_SMSC_DONE));
:2、mHandler实现代码
private Handler mHandler = new Handler() { public void handleMessage(Message msg) { AsyncResult ar; switch (msg.what) { case EVENT_QUERY_SMSC_DONE: ar= (AsyncResult) msg.obj; if (ar.exception != null) { } else { // TODO: handle exception mSmsServiceCenterPref.setSummary((String)ar.result); } } default: break; } } };
原理分析
分析到这大家可能觉得短信中心号码的获取非常简单,但如果要实现该功能仅加上这几行代码,大家可以动手试试,会发现该功能不仅没有实现,甚至还抛出了一个“PhoneFactory.getDefaultPhone must be called from Looper thread”这样的异常。
:1、原因造成该异常的根本原因在通过PhoneFactory类的getDefaultPhone()方法返回一个Phone对象,我们可以进入这个类去看这个函数的实现
public static Phone getDefaultPhone() {
if (sLooper != Looper.myLooper()) { throw new RuntimeException( "PhoneFactory.getDefaultPhone must be called from Looper thread"); }
if (!sMadeDefaults) { throw new IllegalStateException("Default phones haven‘t been made yet!"); } return sProxyPhone[getDefaultSubscription()]; }
该异常由于Looper对象不一样,Looper.myLooper()该对象是指调用getDefaultPhone方法的所在的进程的looper对象,sLooper对象又指的是谁的looper对象,在那赋值?在调用getDefaultPhone时会返回sProxyPhone[getDefaultSubscription()],该数组又是在那赋值的了?
首先,手机启动时会启动phone进程,phone会调用到PhoneApp.java的onCreate方法,在该方法中会调用PhoneFactory.makeDefaultPhones方法。然后,在makeDefaultPhones函数中,会做两件事情,一是初始化sLooper,二是给phone sProxyPhone[]数组赋值。以下是该方法的核心实现:
if (!sMadeDefaults) { sLooper = Looper.myLooper(); sContext = context; if (sLooper == null) { throw new RuntimeException( "PhoneFactory.makeDefaultPhone must be called from Looper thread"); } sProxyPhone = new PhoneProxy[numPhones]; for (int i = 0; i < numPhones; i++) { int phoneType = getPhoneType(networkMode[i]); Log.i(LOG_TAG, "get Phone Type:"+ phoneType); if (phoneType == Phone.PHONE_TYPE_GSM) { sProxyPhone[i] = new PhoneProxy(new GSMPhone(context, sCommandsInterface[i], sPhoneNotifier, false, dct, i)); Log.i(LOG_TAG, "Creating GSMPhone"); } else if (phoneType == Phone.PHONE_TYPE_CDMA) { sProxyPhone[i] = new PhoneProxy(new CDMAPhone(context, sCommandsInterface[i], sPhoneNotifier, false, dct, i)); Log.i(LOG_TAG, "Creating CDMAPhone"); } }
最后,通过前面可以看出sLooper对象是属于phone进程,而Looper.myLooper()是属于调用getDefaultPhone函数的进程,如果调用getDefaultPhone函数的进程和phone进程没有在一个进程空间中执行,则他们的looper对象肯定是不相同的,那么肯定就会抛出“PhoneFactory.getDefaultPhone must be called from Looper thread”这个异常。
:2、解决方法首先在需要在调用getDefaultPhone函数获取phone对象的进程manifest中声明一下代码
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.android.mms" android:sharedUserId="android.uid.phone">
并将具体的Activity声明以下属性:
android:process="com.android.phone"
小结
短信中心号码的读取,这里只讲述了应用层最最基本的实现以及一些注意事项,其最最核心的实现还是落实到了中间层和Ril层.参考类Ril.java、PhoneBase.java。