文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Android开发中使用mms模块收发单卡和双卡短信的教程

2022-06-06 08:58

关注

一.信息发送:
com.android.mms.data.WorkingMessage.java 类
send()函数: 


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,也就是实质内容。 确保文字已拷贝。确保。 根据消息分类,如果是短信直接起一个线程,跑preSendSmsWorker函数,发送短信;如果是彩信,先跑prapareForSave确保文本信息,然后起一个线程,单独跑sendMmsWorker函数。不管是短信还是彩信,起了那个worker函数之一就算发送信息成功了。 最后修改Recipient cache, 重置标志位,过程就结束了。
     短信发送先调用preSendSmsWorker函数,在preSendSmsWorker函数中又起了sendSmsWorker函数。


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


 SmsMessageSender.java类,在mms/transaction下面,实现了MessageSender接口,这个接口sendMessage并返回boolean的值。若发送的是mms,返回true。若发送的是sms,返回false。

当然,对于单卡手机和双卡双待手机的短信发送流程是有区别的(短信接收的流程是相同的,相对流程也比较简洁),关于具体的流程还是直接用UML图来说明更为直接:
信息发送与接收时序图:
发送短信
   单卡手机短信发送的时序图如图所示:

2016222143844109.png (999×533)

双卡手机短信发送的时序图则如下图所示:

2016222143920697.png (1018×533)

二.短信的接收
    信息的接收工作是由底层来完成的,当有一个 新的信息时底层完成接收后会以Intent的方式来通知上层应用,信息的相关内容也包含在Intent当中,Android所支持的信息Intent都定义在android.provider.Telephony.Intents里面。
   
     短信接收,对于上层应用程序来讲就是要处理广播事件SMS_RECEIVED_ACTION,它是由Frameworks发出告诉上层有新的SMS已收到。在Mms中,是由PrivilegedSmsReceiver来处理,它收到SMS_RECEIVED_ACTION(android.provider.Telephony.Intents.SMS_RECEIVED_ACTION=”android.provider.Telephony.SMS_RECEIVED”)后会启动SmsReceiverService来做具体的处理。
SmsReceiverService会先检查短信的类型,如果是Class0短信,直接在GUI中显示,不做任何其他的处理,也即不会存储到数据库中,也不会在Notification Bar中做Notification。
     对于其他短信,会进行替换现有的消息,或是当作新消息插入。原则就是如果在数据库中已有的短信中,与新来的短信的原始地址和协议标识都一样,那么就把其替换成新进的短信,否则就当作新短信插入。
具体的替换流程:先用新进的短信生成一个ContentValues,再用短信的地址和协议标识当作条件到数据库中去查询,如果查到了,就替换,否则就存储。
存储的流程,也是先生成一个CotentValues,然后取出短信的Thread Id和地址,地址要与联系人数据库同步一下,以保证是能识别的地址。如果Thread Id不是合法的,那么就用同步过的地址尝试重新生成Thread Id,尝试5次。然后把刷新过的Thread Id放到ContentValues中,把ContentValues插入到数据库中。如果设置为把信息存储到SIM卡,还要调用SmsManager把信息拷贝到SIM卡上。计算短信的大小,并更新至数据库。删除过期的短信,和超过数量限制的短信,然后返回插入后得到的短信Uri。
最后,对于替换或插入的短信,用Uri去StatusBar做Notification。
GUI在刷新列表时也能得到新短信,因为短信已经被存储到数据库中。

短信接收的时序图如图所示:

2016222144134565.jpg (488×295)

三、双卡双待手机解析短信异常分析及解决
由于是双SIM卡,而且两个卡槽支持的运营商或者网络制式不一定相同,比如一个卡槽支持WCDMA,另一个却只支持GSM ,导致用正常方法解析短信很容易遇到异常。
这里先看下解决方案,这里需要以反射的方式解析不同类型的短信,并且对于不同机型,需对应地进行调整适配:

获取短信信息,注意:为解决双卡双待手机解析短信异常问题,使用Java反射机制,优先解析GSM类型的短信,假如解析失败才按CDMA类型的短信进行解析)


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双卡双待手机 
            message = createFromPduGsm((byte[]) messages[i]); 
          } else if ("CDMA".equals(intent.getStringExtra("from"))) { // 适配MOTO XT800双卡双待手机 
            message = createFromPduCdma((byte[]) messages[i]); 
          } else { 
            message = SmsMessage.createFromPdu((byte[]) messages[i]); // 系统默认的解析短信方式 
          } 
          if (null == message) { // 解决双卡双待类型手机解析短信异常问题 
            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); // 解决双卡双待手机解析短信异常问题 
      } catch (Error er) { 
        er.printStackTrace(); 
        msgs = getSmsMessageByReflect(intent); // 解决双卡双待手机解析短信异常问题 
      } 
    } 
    return msgs; 
  } 

反射方式获取短信


 
  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反射机制解析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; 
    } 
  } 
您可能感兴趣的文章:Android编程之非调用系统界面实现发送彩信的方法(MMS)Android Mms之:短信发送流程(图文详解)Android Mms之:接收信息流程(图文详解)Android Mms之:草稿管理的应用Android Mms之:深入MMS支持Android Mms之:联系人管理的应用分析Android Mms之:深入理解对话列表管理Android Mms之:对话与联系人关联的总结详解Android Mms之:深入理解ComposeAndroid Mms之:PDU的使用详解简单掌握Android开发中彩信的发送接收及其附件的处理


阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-移动开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯