概述
SMS(Short Messaging Service), 即我们经常使用的短信服务。它是一种存储和转发服务。也就是说,短消息并不是直接从发送人发送到接收人,而始终通过 SMS 中心进行转发的。如果接收人处于未连接状态(可能电话已关闭),则消息将在接收人再次连接时发送。
短信长度一般为140个字节,70个字符。既然我们经常使用短信,那么今天我们就来分析一下短信的实现。
说明:本文主要根据android应用层的sms代码来进行分析的,并结合了SMSpopup.有不到位的地方欢迎指正并补充
短信结构
_id // 短消息序号
thread_id // 对话的序号
address // 收件人
person //
date // 日期
protocol // 协议
read // 是否阅读
status // 状态
type // 类型 (收发)
reply_path_present //
subject // 主题
body // 短消息内容
service_center // 服务中心
相关类图
上面几个类是涉及到sms的部分类的类图,其中涉及到键盘锁定状态、指示灯提示、通话状态等的判断,从而决定短信来的时候该怎么样去处理和提示。
短信接收
先来看看短信的接收,在android中,短信的接收需要在manifest.xml中配置广播接收器,如下:
来看下时序图:
/**
* 接收到短信的处理放在该service中去进行处理
*/
public static void beginStartingService(Context context, Intent intent) {
synchronized (mStartingServiceSync) {
if (Log.DEBUG) Log.v("SMSReceiverService: beginStartingService()");
if (mStartingService == null) {
//电源管理,决定是否亮指示灯、键盘、屏幕等
PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
mStartingService = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
Log.LOGTAG+".SmsReceiverService");
mStartingService.setReferenceCounted(false);
}
mStartingService.acquire();
context.startService(intent);
}
}
在接收到短信的时候,需要有一些提示,比如指示灯点亮、屏幕点亮、键盘点亮等。这个主要通过PowerManager来控制。关于PowerManager,可以参看我之间写的一篇文章:http://blog.csdn.net/xieqibao/article/details/6562256
ServiceHandler的handleMessage方法中处理消息,判断消息的类型是mms、sms,在handleSmsReceived中处理接收到的sms短信
/**
* 处理接收到的短信息
*/
private void handleSmsReceived(Intent intent) {
Bundle bundle = intent.getExtras();
if (bundle != null) {
//获得消息
SmsMessage[] messages = SmsPopupUtils.getMessagesFromIntent(intent);
if (messages != null) {
notifyMessageReceived(new SmsMmsMessage(context, messages,System.currentTimeMillis()));
}
}
}
下面方法中主要是从pdu中获取信息,关于pdu的详细信息可以参考:http://wenku.baidu.com/view/d0d0093e0912a216147929b1.html
/**
* 从包装了SMS_RECEIVED_ACTION的intent中获取pdu信息
*/
public static final SmsMessage[] getMessagesFromIntent(Intent intent) {
Object[] messages = (Object[]) intent.getSerializableExtra("pdus");
if (messages == null) {
return null;
}
if (messages.length == 0) {
return null;
}
byte[][] pduObjs = new byte[messages.length][];
for (int i = 0; i < messages.length; i++) {
pduObjs[i] = (byte[]) messages[i];
}
byte[][] pdus = new byte[pduObjs.length][];
int pduCount = pdus.length;
SmsMessage[] msgs = new SmsMessage[pduCount];
for (int i = 0; i < pduCount; i++) {
pdus[i] = pduObjs[i];
msgs[i] = SmsMessage.createFromPdu(pdus[i]);
}
return msgs;
}
最终获取到短信息后决定该怎么去进行展示,在smspopup中,是通过弹窗的形式,把短信息显示在activity中
/**
* 获取短信息并加入到view上
*/
private void setupMessages(Bundle b, boolean newIntent) {
// Store bundle
bundle = b;
// 从bundle中获取短消息
SmsMmsMessage message = new SmsMmsMessage(getApplicationContext(), bundle);
mSmsPopups.addMessage(message);
if (!newIntent) {
// TODO: move off UI thread
mSmsPopups.addMessages(
SmsPopupUtils.getUnreadMessages(this, message.getMessageId()));
}
mSmsPopups.refreshPrivacy();
}
短信发送
/**
* 回复短信息
*
*/
public boolean replyToMessage(StringquickReply) {
//首先要标记短信为已读
setMessageRead();
// 发送新的短信息
SmsMessageSender sender =
new SmsMessageSender(context, newString[] {fromAddress}, quickReply, getThreadId());
return sender.sendMessage();
}
/**
* 发送短信息入口
* @return
*/
@SuppressWarnings("deprecation")
public boolean sendMessage() {
if (!(mThreadId > 0)) {
return false;
}
//如果消息文本为空,那就不发消息了
if ((mMessageText == null) ||(mNumberOfDests == 0)) {
return false;
}
//获得短信管理器
SmsManager smsManager =SmsManager.getDefault();
for (int i = 0; i < mNumberOfDests; i++){
//按照短信息允许的最大字数来拆分短信
ArrayList messages =smsManager.divideMessage(mMessageText);
int messageCount = messages.size();
ArrayListdeliveryIntents = new ArrayList(messageCount);
ArrayListsentIntents = new ArrayList(messageCount);
// 140个字节,70字符。
if (splitMessage) {
for (int j = 0; j < messageCount;j++) {
Uri uri = null;
try
//把短信息加入到provider中
uri =addMessage(mContext.getContentResolver(), mDests[i], messages.get(j),
null, mTimestamp,requestDeliveryReport, mThreadId);
} catch (SQLiteException e) {
// TODO: show error here
//SqliteWrapper.checkSQLiteException(mContext, e);
}
PendingIntent deliveryReportIntent =null;
if (requestDeliveryReport) {
deliveryReportIntent =
PendingIntent.getBroadcast(mContext, 0,
newIntent(MESSAGING_STATUS_RECEIVED_ACTION, uri)
.setClassName(MESSAGING_PACKAGE_NAME, MESSAGING_STATUS_CLASS_NAME), 0);
}
PendingIntent sentIntent =
PendingIntent.getBroadcast(mContext, 0,
newIntent(SmsReceiverService.MESSAGE_SENT_ACTION, uri)
.setClass(mContext,SmsReceiver.class), 0);
smsManager.sendTextMessage(
mDests[i], mServiceCenter,messages.get(j), sentIntent, deliveryReportIntent);
}
}
return false;
}
总结
1. sms短信主要涉及到短信的发送、接收、提示,以及短信的本地保存等,涉及到的点相对还是比较多的。了解下还是很有必要的。