亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Android開發(fā)中使用mms模塊收發(fā)單卡和雙卡短信的教程

 更新時間:2016年02月22日 14:45:04   作者:時之沙  
這篇文章主要介紹了Android開發(fā)中使用mms模塊收發(fā)單卡和雙卡短信的教程,文中舉了MOTO XT800手機(jī)(估計已經(jīng)落伍很久了--)的例子來說明如何解決雙卡雙待時的短信異常問題,需要的朋友可以參考下

一.信息發(fā)送:
com.android.mms.data.WorkingMessage.java 類
send()函數(shù): 

public void send() {   
 
  ......  
 
  if (requiresMms() || addressContainsEmailToMms(conv, msgTxt)) {   
 
    // 彩信   
 
    slideshow.prepareForSend(); 
 
    new Thread(new Runnable() {   
 
      public void run() {   
 
        sendMmsWorker(conv, mmsUri, persister, slideshow, sendReq);   
 
      }   
 
    }).start();   
 
  } else {   
 
    // 短信   
 
    new Thread(new Runnable() {   
 
      public void run() {   
 
        preSendSmsWorker(conv, msgText);   
 
      }   
 
    }).start();   
 
  ...... 
 
}  


prapareForSend(). 先確保有slidshow,也就是實質(zhì)內(nèi)容。 確保文字已拷貝。確保標(biāo)題。 根據(jù)消息分類,如果是短信直接起一個線程,跑preSendSmsWorker函數(shù),發(fā)送短信;如果是彩信,先跑prapareForSave確保文本信息,然后起一個線程,單獨跑sendMmsWorker函數(shù)。不管是短信還是彩信,起了那個worker函數(shù)之一就算發(fā)送信息成功了。 最后修改Recipient cache, 重置標(biāo)志位,過程就結(jié)束了。
     短信發(fā)送先調(diào)用preSendSmsWorker函數(shù),在preSendSmsWorker函數(shù)中又起了sendSmsWorker函數(shù)。

private void sendSmsWorker(String msgText, String semiSepRecipients, long threadId) { 
 
......  
 
MessageSender sender = new SmsMessageSender(mContext, dests, msgText, threadId); 
 
sender.sendMessage(threadId); 
 
......    
 
} 


 SmsMessageSender.java類,在mms/transaction下面,實現(xiàn)了MessageSender接口,這個接口sendMessage并返回boolean的值。若發(fā)送的是mms,返回true。若發(fā)送的是sms,返回false。

當(dāng)然,對于單卡手機(jī)和雙卡雙待手機(jī)的短信發(fā)送流程是有區(qū)別的(短信接收的流程是相同的,相對流程也比較簡潔),關(guān)于具體的流程還是直接用UML圖來說明更為直接:
信息發(fā)送與接收時序圖:
發(fā)送短信
   單卡手機(jī)短信發(fā)送的時序圖如圖所示:

2016222143844109.png (999×533)

雙卡手機(jī)短信發(fā)送的時序圖則如下圖所示:

2016222143920697.png (1018×533)

二.短信的接收
    信息的接收工作是由底層來完成的,當(dāng)有一個 新的信息時底層完成接收后會以Intent的方式來通知上層應(yīng)用,信息的相關(guān)內(nèi)容也包含在Intent當(dāng)中,Android所支持的信息Intent都定義在android.provider.Telephony.Intents里面。
   
     短信接收,對于上層應(yīng)用程序來講就是要處理廣播事件SMS_RECEIVED_ACTION,它是由Frameworks發(fā)出告訴上層有新的SMS已收到。在Mms中,是由PrivilegedSmsReceiver來處理,它收到SMS_RECEIVED_ACTION(android.provider.Telephony.Intents.SMS_RECEIVED_ACTION=”android.provider.Telephony.SMS_RECEIVED”)后會啟動SmsReceiverService來做具體的處理。
SmsReceiverService會先檢查短信的類型,如果是Class0短信,直接在GUI中顯示,不做任何其他的處理,也即不會存儲到數(shù)據(jù)庫中,也不會在Notification Bar中做Notification。
     對于其他短信,會進(jìn)行替換現(xiàn)有的消息,或是當(dāng)作新消息插入。原則就是如果在數(shù)據(jù)庫中已有的短信中,與新來的短信的原始地址和協(xié)議標(biāo)識都一樣,那么就把其替換成新進(jìn)的短信,否則就當(dāng)作新短信插入。
具體的替換流程:先用新進(jìn)的短信生成一個ContentValues,再用短信的地址和協(xié)議標(biāo)識當(dāng)作條件到數(shù)據(jù)庫中去查詢,如果查到了,就替換,否則就存儲。
存儲的流程,也是先生成一個CotentValues,然后取出短信的Thread Id和地址,地址要與聯(lián)系人數(shù)據(jù)庫同步一下,以保證是能識別的地址。如果Thread Id不是合法的,那么就用同步過的地址嘗試重新生成Thread Id,嘗試5次。然后把刷新過的Thread Id放到ContentValues中,把ContentValues插入到數(shù)據(jù)庫中。如果設(shè)置為把信息存儲到SIM卡,還要調(diào)用SmsManager把信息拷貝到SIM卡上。計算短信的大小,并更新至數(shù)據(jù)庫。刪除過期的短信,和超過數(shù)量限制的短信,然后返回插入后得到的短信Uri。
最后,對于替換或插入的短信,用Uri去StatusBar做Notification。
GUI在刷新列表時也能得到新短信,因為短信已經(jīng)被存儲到數(shù)據(jù)庫中。

短信接收的時序圖如圖所示:

2016222144134565.jpg (488×295)

三、雙卡雙待手機(jī)解析短信異常分析及解決
由于是雙SIM卡,而且兩個卡槽支持的運營商或者網(wǎng)絡(luò)制式不一定相同,比如一個卡槽支持WCDMA,另一個卻只支持GSM ,導(dǎo)致用正常方法解析短信很容易遇到異常。
這里先看下解決方案,這里需要以反射的方式解析不同類型的短信,并且對于不同機(jī)型,需對應(yīng)地進(jìn)行調(diào)整適配:

獲取短信信息,注意:為解決雙卡雙待手機(jī)解析短信異常問題,使用Java反射機(jī)制,優(yōu)先解析GSM類型的短信,假如解析失敗才按CDMA類型的短信進(jìn)行解析)

public static SmsMessage[] getSmsMessage(Intent intent) { 
    SmsMessage[] msgs = null; 
    Object messages[] = (Object[]) intent.getSerializableExtra("pdus"); 
    int len = 0; 
    if (null != messages && (len = messages.length) > 0) { 
      msgs = new SmsMessage[len]; 
      try { 
        for (int i = 0; i < len; i++) { 
          SmsMessage message = null; 
          if ("GSM".equals(intent.getStringExtra("from"))) { // 適配MOTO XT800雙卡雙待手機(jī) 
            message = createFromPduGsm((byte[]) messages[i]); 
          } else if ("CDMA".equals(intent.getStringExtra("from"))) { // 適配MOTO XT800雙卡雙待手機(jī) 
            message = createFromPduCdma((byte[]) messages[i]); 
          } else { 
            message = SmsMessage.createFromPdu((byte[]) messages[i]); // 系統(tǒng)默認(rèn)的解析短信方式 
          } 
          if (null == message) { // 解決雙卡雙待類型手機(jī)解析短信異常問題 
            message = createFromPduGsm((byte[]) messages[i]); 
            if (null == message) { 
              message = createFromPduCdma((byte[]) messages[i]); 
            } 
          } 
          if (null != message) { 
            msgs[i] = message; 
          } 
        } 
      } catch (Exception e) { 
        e.printStackTrace(); 
        msgs = getSmsMessageByReflect(intent); // 解決雙卡雙待手機(jī)解析短信異常問題 
      } catch (Error er) { 
        er.printStackTrace(); 
        msgs = getSmsMessageByReflect(intent); // 解決雙卡雙待手機(jī)解析短信異常問題 
      } 
    } 
    return msgs; 
  } 

反射方式獲取短信

/** 
   * 使用Java反射機(jī)制獲取短信信息(解決雙卡雙待手機(jī)解析短信異常問題,優(yōu)先解析GSM類型的短信,假如解析失敗才按CDMA類型的短信進(jìn)行解析) 
   * 
   * @param intent 
   * @return 
   */ 
  private static SmsMessage[] getSmsMessageByReflect(Intent intent) { 
    SmsMessage[] msgs = null; 
    Object messages[] = (Object[]) intent.getSerializableExtra("pdus"); 
    int len = 0; 
    if (null != messages && (len = messages.length) > 0) { 
      msgs = new SmsMessage[len]; 
      try { 
        for (int i = 0; i < len; i++) { 
          SmsMessage message = createFromPduGsm((byte[]) messages[i]); 
          if (null == message) { 
            message = createFromPduCdma((byte[]) messages[i]); 
          } 
          if (null != message) { 
            msgs[i] = message; 
          } 
        } 
      } catch (SecurityException e) { 
        e.printStackTrace(); 
      } catch (IllegalArgumentException e) { 
        e.printStackTrace(); 
      } catch (ClassNotFoundException e) { 
        e.printStackTrace(); 
      } catch (NoSuchMethodException e) { 
        e.printStackTrace(); 
      } catch (IllegalAccessException e) { 
        e.printStackTrace(); 
      } catch (InvocationTargetException e) { 
        e.printStackTrace(); 
      } catch (InstantiationException e) { 
        e.printStackTrace(); 
      } 
    } 
    return msgs; 
  } 

通過Java反射機(jī)制解析GSM類型的短信:

private static SmsMessage createFromPduGsm(byte[] pdu) throws SecurityException, IllegalArgumentException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { 
    return createFromPdu(pdu, "com.android.internal.telephony.gsm.SmsMessage"); 
  } 

解析CDMA類型的短信

private static SmsMessage createFromPduCdma(byte[] pdu) throws SecurityException, IllegalArgumentException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { 
  return createFromPdu(pdu, "com.android.internal.telephony.cdma.SmsMessage"); 
} 

解析GSM或者CDMA類型的短信

private static SmsMessage createFromPdu(byte[] pdu, String className) throws ClassNotFoundException, SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException, InstantiationException { 
    Class<?> clazz = Class.forName(className); 
    Object object = clazz.getMethod("createFromPdu", byte[].class).invoke(clazz.newInstance(), pdu); 
    if (null != object) { 
      Constructor<?> constructor = SmsMessage.class.getDeclaredConstructor(Class.forName("com.android.internal.telephony.SmsMessageBase")); 
      constructor.setAccessible(true); 
      return (SmsMessage) constructor.newInstance(object); 
    } else { 
      return null; 
    } 
  } 

相關(guān)文章

  • Android編程開發(fā)之多點觸摸(Multitouch)實現(xiàn)方法

    Android編程開發(fā)之多點觸摸(Multitouch)實現(xiàn)方法

    這篇文章主要介紹了Android編程開發(fā)之多點觸摸(Multitouch)實現(xiàn)方法,結(jié)合實例形式詳細(xì)分析了Android多點觸摸的相關(guān)實現(xiàn)步驟與操作技巧,需要的朋友可以參考下
    2016-08-08
  • Android中AOP(面向切向編程)的深入講解

    Android中AOP(面向切向編程)的深入講解

    這篇文章主要給大家介紹了關(guān)于Android中AOP(面向切向編程)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-09-09
  • Android操作SQLite基本用法

    Android操作SQLite基本用法

    這篇文章主要介紹了Android操作SQLite基本用法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2021-12-12
  • Android實現(xiàn)通過手勢控制圖片大小縮放的方法

    Android實現(xiàn)通過手勢控制圖片大小縮放的方法

    這篇文章主要介紹了Android實現(xiàn)通過手勢控制圖片大小縮放的方法,結(jié)合實例形式分析了Android控制圖片縮放的原理、實現(xiàn)步驟與相關(guān)操作技巧,需要的朋友可以參考下
    2016-10-10
  • Android深入探究自定義View之嵌套滑動的實現(xiàn)

    Android深入探究自定義View之嵌套滑動的實現(xiàn)

    什么是嵌套滑動?當(dāng)我們向下滑動時,首先是外部的布局向下滑動,然后才是內(nèi)部的RecyclerView滑動,向上滑動也是如此。這就是嵌套滑動的效果
    2021-11-11
  • Android 存儲路徑選擇方法

    Android 存儲路徑選擇方法

    今天小編就為大家分享一篇Android 存儲路徑選擇方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-08-08
  • Android 選擇相冊照片并返回功能的實現(xiàn)代碼

    Android 選擇相冊照片并返回功能的實現(xiàn)代碼

    這篇文章主要介紹了Android 從相冊中選擇照片并返回功能,需要的朋友可以參考下
    2018-03-03
  • Android實現(xiàn)閃屏頁效果

    Android實現(xiàn)閃屏頁效果

    這篇文章主要為大家詳細(xì)介紹了Android實現(xiàn)閃屏頁效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-11-11
  • ImageView的屬性android:scaleType的作用分析

    ImageView的屬性android:scaleType的作用分析

    本篇文章是對ImageView的屬性android:scaleType的作用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-06-06
  • Android應(yīng)用程序中讀寫txt文本文件的基本方法講解

    Android應(yīng)用程序中讀寫txt文本文件的基本方法講解

    這篇文章主要介紹了Android應(yīng)用程序中讀寫txt文本文件的基本方法講解,基本上依靠context.openFileInput()和context.openFileOutput()兩個方法為主,需要的朋友可以參考下
    2016-04-04

最新評論