Android短彩信收发流程(Framework)

2024-03-31 09:38

本文主要是介绍Android短彩信收发流程(Framework),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

短信部分:

短信的发送,Framework部分从SmsManager的sendTextMessage(普通短信,无分段)与sendMultipartTextMessage(长短信,已分段)开始,一直到RIL。主要涉及到的类有。SmsManager、Isms、IccSmsInterfaceManager、SMSDispatcher、ImsSMSDispatcher、GsmSMSDispatcher、CdmaSMSDispatcher、SmsMessage、SmsTracker、CommandsInterface、BaseCommands、RIL、LocalSocket、RilRequest、Parcel等。


SmsManager->RIL:
sendTextMessage和sendMultipartTextMessage的过程其实差不多,无非是sendMultipartTextMessage对长短信进行了一下处理,为每一个part设置了SmsHeader。ImsSMSDispatcher中包含了两种不同的SmsDispatcher,IccSmsInterfaceManager调用SmsDispatcher中各个方法时,会根据手机类型的不同,调用不同SmsDispatcher的相应方法。大致过程为:1、计算编码方式;2、对每一个分段构造SmsHeader、SubmitPdu、SmsTracker,并设置RadioTechnologyFamily;3、利用SmsTracker构造当消息正常发送后调用的Message对象;4、调用与手机卡类型有关的RIL中的相应方法发送。


RIL中发送:
sendImsGsmSms、sendSms、sendCdmaSms、sendImsCdmaSms这四个方法基本流程是大致相同的,都是先构造RILRequest,再调用send方法发送。区别就在于不同的方法在获得RILRequest时传入的请求类型不同,构造出来的pdu结构不同,以及两个Ims方法需要先往RILRequest中写入一个数字。


发送后的处理:
已发送状态修改(sentIntent的调用过程)RILReceiver->SMSDispatcher:
在RIL的构造方法中,创建了一个RILReceiver对象,该对象实现了Runnable接口。它的作用是不停地从下层获取数据,分析类型,转到相应SMSDispatcher处理。
当短信通过LocalSocket发送出去后,由于是主动请求,下层会传上来一个响应,其基本类型为RESPONSE_SOLICITED,转入processSolicited处理。在该方法中,对于短信发送的处理,均调用responseSMS方法获取SmsResponse对象。(前面四种方式对应的事件类型为sendSMS-RIL_REQUEST_SEND_SMS、 sendCdmaSms-RIL_REQUEST_CDMA_SEND_SMS、sendImsGsmSms&sendImsCdmaSms-RIL_REQUEST_IMS_SEND_SMS)
然后创建AsyncResult对象,将SmsResponse对象与SmsTracker对象(SMSRequest中的)传入,更新之前在各个SMSDispatcher中创建的消息正常发送后调用的Message中的obj,设为创建的AsyncResult对象。将消息发送至相应SmsDispatcher处理。


已发送状态修改(sentIntent的调用过程)SMSDispatcher.handleMessage:
由于之前在构造Message对象时设置了what为SEND_SMS_COMPLETE,因此,此处调用handleSendComplete方法进行短信发送的后续处理。若需要发送报告,将SmsTracker加入deliveryPendingList中,这样后面收到发送报告时,能够从该列表中取出进行处理(具体参见应用层部分发送报告的处理)。


deliveryIntent(发送报告)RIL->SMSDispatcher:
对于GSM,在构造方法中,将GsmSMSDispatcher注册为RIL接收到发送报告时该事件的接收者,并设置消息类型为EVENT_NEW_SMS_STATUS_REPORT,相应的Registrant类为mSmsStatusRegistrant。
而对于CDMA,则设置消息类型为EVENT_NEW_SMS,相应的Registrant类为mCdmaSMSRegistrant。
当RIL收到底层传来的发送报告后(过程与已发送状态修改相同),会产生一个基本类型为RESPONSE_UNSOLICITED,转入processUnsolicited处理。
对于GSM,其事件类型为RIL_UNSOL_RESPONSE_NEW_SMS_STATUS_REPORT。先调用responseString从Parcel中获取数据,再调用mSmsStatusRegistrant的notifyRegistrant方法设置消息类型(what属性为EVENT_NEW_SMS_STATUS_REPORT)并转到SMSDispatcher进行处理。
而CDMA,事件类型为RIL_UNSOL_RESPONSE_CDMA_NEW_SMS。处理过程与GSM大致相同,只是从Parcel中获取数据是调用responseCdmaSms方法获取SmsMessage对象。然后调用mCdmaSMSRegistrant的notifyRegistrant方法设置消息类型(what属性为EVENT_NEW_SMS)并转到SMSDispatcher进行处理。
deliveryIntent(发送报告)SMSDispatcher:
对于GSM,直接调用handleStatusReport方法处理。从传入的AsyncResult对象中获取SmsMessage进而获取SmsTracker的索引,从deliveryPendingList中取出SmsTracker,发送deliveryIntent并发送消息确认。
对于CDMA,在handleMessage中转到EVENT_NEW_SMS。调用dispatchMessage进行消息的分发。


短信接收RIL->SMSDispatcher:
GSM,在构造方法中,将GsmSMSDispatcher注册为RIL接收短信时该事件的接收者,并设置消息类型为EVENT_NEW_SMS,相应的Registrant类为mSMSRegistrant。
CDMA,与发送报告相同,消息类型为EVENT_NEW_SMS,相应的Registrant类为mCdmaSMSRegistrant。
RILReceiver接收到短信后,会转到processUnsolicited进行处理。
对于GSM,其事件类型为RIL_UNSOL_RESPONSE_NEW_SMS。先调用responseString从Parcel中获取数据,再使用newFromCMT方法获取SmsMessage对象,最后调用mSMSRegistrant的notifyRegistrant方法设置消息类型(what属性为EVENT_NEW_SMS)并转到SMSDispatcher进行处理。
而CDMA,事件类型为RIL_UNSOL_RESPONSE_CDMA_NEW_SMS。过程与接收报告相同。
短信接收SMSDispatcher:
GSM,首先获取SmsHeader。
如果SmsHeader或SmsHeader.concatRef均不为空,说明是长短信,则调用processMessagePart将短信分段存入raw表,待所有分段都收到后,将其组装。然后根据端口的不同,按照彩信通知(WapPushOverSms的dispatchWapPdu方法)、指定端口的彩信(dispatchPortAddressedPdus)、长短信(dispatchPdus)进行分发处理。
若SmsHeader或SmsHeader.concatRef有一个为空,说明不是长短信。然后根据端口的不同,按照彩信通知(WapPushOverSms的dispatchWapPdu方法)、指定端口的彩信(dispatchPortAddressedPdus)、普通短信(dispatchPdus)进行分发处理。
GSM和CDMA的短信接收有很大一部分是相同的,只是CDMA由于标准定义的不同,需要进行一些其他的处理。
CDMA没有类似GSM中的User Data Header(UDA)来存储一些额外信息,而是使用一个16位整数CDMA TELESERVICE来确定消息的类型。
如果其为TELESERVICE_WAP,说明是CDMA彩信通知,则调用processCdmaWapPdu来处理。该方法的作用是将接收到的分段数据先存入raw表。只有当所有分段都接收到后,如果端口是WAP_PUSH的彩信,才调用WapPushOverSms的dispatchWapPdu方法将彩信通知内容解码后通过广播发送到应用层。否则,调用dispatchPortAddressedPdus方法将其发送到指定端口(sms://localhost:port)。
另外,CDMA中提供了一种cmas消息。如果是这种消息,当收到时,立即调用dispatchBroadcastPdus方法发送紧急广播。


彩信接收:
在Android中,彩信的接收分为两部分。彩信通知通过短信的方式接收,如上文所述。彩信数据的下载在应用层中处理

这篇关于Android短彩信收发流程(Framework)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/864030

相关文章

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Spring Boot分层架构详解之从Controller到Service再到Mapper的完整流程(用户管理系统为例)

《SpringBoot分层架构详解之从Controller到Service再到Mapper的完整流程(用户管理系统为例)》本文将以一个实际案例(用户管理系统)为例,详细解析SpringBoot中Co... 目录引言:为什么学习Spring Boot分层架构?第一部分:Spring Boot的整体架构1.1

nodejs打包作为公共包使用的完整流程

《nodejs打包作为公共包使用的完整流程》在Node.js项目中,打包和部署是发布应用的关键步骤,:本文主要介绍nodejs打包作为公共包使用的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言一、前置准备二、创建与编码三、一键构建四、本地“白嫖”测试(可选)五、发布公共包六、常见踩坑提醒

Ubuntu向多台主机批量传输文件的流程步骤

《Ubuntu向多台主机批量传输文件的流程步骤》:本文主要介绍在Ubuntu中批量传输文件到多台主机的方法,需确保主机互通、用户名密码统一及端口开放,通过安装sshpass工具,准备包含目标主机信... 目录Ubuntu 向多台主机批量传输文件1.安装 sshpass2.准备主机列表文件3.创建一个批处理脚

一个Java的main方法在JVM中的执行流程示例详解

《一个Java的main方法在JVM中的执行流程示例详解》main方法是Java程序的入口点,程序从这里开始执行,:本文主要介绍一个Java的main方法在JVM中执行流程的相关资料,文中通过代码... 目录第一阶段:加载 (Loading)第二阶段:链接 (Linking)第三阶段:初始化 (Initia

Git打标签从本地创建到远端推送的详细流程

《Git打标签从本地创建到远端推送的详细流程》在软件开发中,Git标签(Tag)是为发布版本、标记里程碑量身定制的“快照锚点”,它能永久记录项目历史中的关键节点,然而,仅创建本地标签往往不够,如何将其... 目录一、标签的两种“形态”二、本地创建与查看1. 打附注标http://www.chinasem.cn

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

在Android中使用WebView在线查看PDF文件的方法示例

《在Android中使用WebView在线查看PDF文件的方法示例》在Android应用开发中,有时我们需要在客户端展示PDF文件,以便用户可以阅读或交互,:本文主要介绍在Android中使用We... 目录简介:1. WebView组件介绍2. 在androidManifest.XML中添加Interne

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

MyBatis分页查询实战案例完整流程

《MyBatis分页查询实战案例完整流程》MyBatis是一个强大的Java持久层框架,支持自定义SQL和高级映射,本案例以员工工资信息管理为例,详细讲解如何在IDEA中使用MyBatis结合Page... 目录1. MyBATis框架简介2. 分页查询原理与应用场景2.1 分页查询的基本原理2.1.1 分