前言

短信中心号码概念

短信中心号码类似信息服务器,如果信息中心号码不正确,短消息是无法发送成功的,各个地区都有自己的信息中心号码,其中移动以+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。