天天看点

Android 之 sms 短信

  • 概述

    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      // 服务中心  

    相关类图

    Android 之 sms 短信

    上面几个类是涉及到sms的部分类的类图,其中涉及到键盘锁定状态、指示灯提示、通话状态等的判断,从而决定短信来的时候该怎么样去处理和提示。

    短信接收

    先来看看短信的接收,在android中,短信的接收需要在manifest.xml中配置广播接收器,如下:     

<receiver android:name=".SmsReceiver">
        <intent-filter>
            <action android:name="android.provider.Telephony.SMS_RECEIVED" />
        </intent-filter>
   </receiver>
           
  • 来看下时序图:
    Android 之 sms 短信
    /**
           *  接收到短信的处理放在该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();
      }
               
    短信发送
    Android 之 sms 短信
/**
   * 回复短信息
   *
   */
  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<String> messages =smsManager.divideMessage(mMessageText);
      int messageCount = messages.size();
      ArrayList<PendingIntent>deliveryIntents = new ArrayList<PendingIntent>(messageCount);
      ArrayList<PendingIntent>sentIntents = new ArrayList<PendingIntent>(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短信主要涉及到短信的发送、接收、提示,以及短信的本地保存等,涉及到的点相对还是比较多的。了解下还是很有必要的。