Android 之 sms 短信

来源:互联网 发布:八字网站 知乎 编辑:程序博客网 时间:2024/05/29 14:31

    概述

    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中配置广播接收器,如下:     

   <receiver android:name=".SmsReceiver">        <intent-filter>            <action android:name="android.provider.Telephony.SMS_RECEIVED" />        </intent-filter>   </receiver>

    来看下时序图:

     

        /**       *  接收到短信的处理放在该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<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短信主要涉及到短信的发送、接收、提示,以及短信的本地保存等,涉及到的点相对还是比较多的。了解下还是很有必要的。

 


原创粉丝点击