热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android发送短信方法总结

这篇文章主要介绍了Android发送短信方法,结合实例形式较为详细的分析了Android发送短信的原理与具体实现技巧,需要的朋友可以参考下

android API 中提供了SmsManager类处理短信。其中的sendTextMessage(num, null, content, pend, null)函数就是发送,具体介绍如下:

SMS涉及的主要类SmsManager

实现SMS主要用到SmsManager类,该类继承自java.lang.Object类,下面我们介绍一下该类的主要成员。
公有方法:

1、ArrayList divideMessage(String text)

当短信超过SMS消息的最大长度时,将短信分割为几块。
参数:text——初始的消息,不能为空
返回值:有序的ArrayList,可以重新组合为初始的消息

2、static SmsManager getDefault()

获取SmsManager的默认实例。
返回值:SmsManager的默认实例

3、void SendDataMessage(String destinationAddress, String scAddress, short destinationPort, byte[] data,PendingIntent sentIntent, PendingIntent deliveryIntent)

发送一个基于SMS的数据到指定的应用程序端口。
参数:
1)、destinationAddress——消息的目标地址

2)、scAddress——服务中心的地址or为空使用当前默认的SMSC

3)destinationPort——消息的目标端口号

4)、data——消息的主体,即消息要发送的数据

5)、sentIntent——如果不为空,当消息成功发送或失败这个PendingIntent就广播。结果代码是Activity.RESULT_OK表示成功,或RESULT_ERROR_GENERIC_FAILURE、RESULT_ERROR_RADIO_OFF、RESULT_ERROR_NULL_PDU之一表示错误。对应RESULT_ERROR_GENERIC_FAILURE,sentIntent可能包括额外的“错误代码”包含一个无线电广播技术特定的值,通常只在修复故障时有用。
每一个基于SMS的应用程序控制检测sentIntent。如果sentIntent是空,调用者将检测所有未知的应用程序,这将导致在检测的时候发送较小数量的SMS。

6)、deliveryIntent——如果不为空,当消息成功传送到接收者这个PendingIntent就广播。
异常:如果destinationAddress或data是空时,抛出IllegalArgumentException异常。

4、void sendMultipartTextMessage(String destinationAddress, String scAddress, ArrayList parts,ArrayList sentIntents, ArrayList  deliverIntents)

发送一个基于SMS的多部分文本,调用者应用已经通过调用divideMessage(String text)将消息分割成正确的大小。

参数:

1)、destinationAddress——消息的目标地址

2)、scAddress——服务中心的地址or为空使用当前默认的SMSC

3)、parts——有序的ArrayList,可以重新组合为初始的消息

4)、sentIntents——跟SendDataMessage方法中一样,只不过这里的是一组PendingIntent

5)、deliverIntents——跟SendDataMessage方法中一样,只不过这里的是一组PendingIntent
异常:如果destinationAddress或data是空时,抛出IllegalArgumentException异常。

5、void sendTextMessage(String destinationAddress, String scAddress, String text, PendingIntent sentIntent,PendingIntent deliveryIntent)

发送一个基于SMS的文本。参数的意义和异常前面的已存在的一样,不再累述。

常量:

  • public static final int RESULT_ERROR_GENERIC_FAILURE   表示普通错误,值为1(0x00000001)
  • public static final int RESULT_ERROR_NO_SERVICE    表示服务当前不可用,值为4 (0x00000004)
  • public static final int RESULT_ERROR_NULL_PDU   表示没有提供pdu,值为3 (0x00000003)
  • public static final int RESULT_ERROR_RADIO_OFF   表示无线广播被明确地关闭,值为2 (0x00000002)
  • public static final int STATUS_ON_ICC_FREE    表示自由空间,值为0 (0x00000000)
  • public static final int STATUS_ON_ICC_READ  表示接收且已读,值为1 (0x00000001)
  • public static final int STATUS_ON_ICC_SENT   表示存储且已发送,值为5 (0x00000005)
  • public static final int STATUS_ON_ICC_UNREAD  表示接收但未读,值为3 (0x00000003)
  • public static final int STATUS_ON_ICC_UNSENT  表示存储但为发送,值为7 (0x00000007)

 第一:调用系统短信接口直接发送短信;主要代码如下:

 /**
   * 直接调用短信接口发短信
   * 
   * @param phoneNumber
   * @param message
   */
  public void sendSMS(String phoneNumber, String message) {
    // 获取短信管理器
    android.telephony.SmsManager smsManager = android.telephony.SmsManager
        .getDefault();
    // 拆分短信内容(手机短信长度限制)
    List divideCOntents= smsManager.divideMessage(message);
    for (String text : divideContents) {
      smsManager.sendTextMessage(phoneNumber, null, text, sentPI,
          deliverPI);
    }
  }

第二:调起系统发短信功能;主要代码如下:

 /**
   * 调起系统发短信功能
   * @param phoneNumber
   * @param message
   */
  public void doSendSMSTo(String phoneNumber,String message){
    if(PhoneNumberUtils.isGlobalPhoneNumber(phoneNumber)){
      Intent intent = new Intent(Intent.ACTION_SENDTO, Uri.parse("smsto:"+phoneNumber));     
      intent.putExtra("sms_body", message);     
      startActivity(intent);
    }
  }

下面来主要讲解第一种方法,第一种方法可以监控发送状态和对方接收状态使用的比较多。

 处理返回的状态代码如下:

//处理返回的发送状态 
    String SENT_SMS_ACTION = "SENT_SMS_ACTION";
    Intent sentIntent = new Intent(SENT_SMS_ACTION);
    sentPI= PendingIntent.getBroadcast(this, 0, sentIntent,
        0);
    // register the Broadcast Receivers
    this.registerReceiver(new BroadcastReceiver() {
      @Override
      public void onReceive(Context _context, Intent _intent) {
        switch (getResultCode()) {
        case Activity.RESULT_OK:
          Toast.makeText(MainActivity.this,
        "短信发送成功", Toast.LENGTH_SHORT)
        .show();
        break;
        case SmsManager.RESULT_ERROR_GENERIC_FAILURE:
        break;
        case SmsManager.RESULT_ERROR_RADIO_OFF:
        break;
        case SmsManager.RESULT_ERROR_NULL_PDU:
        break;
        }
      }
    }, new IntentFilter(SENT_SMS_ACTION));

    
    //处理返回的接收状态 
    String DELIVERED_SMS_ACTION = "DELIVERED_SMS_ACTION";
    // create the deilverIntent parameter
    Intent deliverIntent = new Intent(DELIVERED_SMS_ACTION);
    deliverPI = PendingIntent.getBroadcast(this, 0,
        deliverIntent, 0);
    this.registerReceiver(new BroadcastReceiver() {
      @Override
      public void onReceive(Context _context, Intent _intent) {
        Toast.makeText(MainActivity.this,
     "收信人已经成功接收", Toast.LENGTH_SHORT)
     .show();
      }
    }, new IntentFilter(DELIVERED_SMS_ACTION));

以下是对以上代码的封装

1、短信的发送

import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.telephony.SmsManager;

import java.util.ArrayList;
import java.util.List;

/**
 * Created by Javen on 2016-03-15.
 */
public class SMSMethod {
  private static SMSMethod mSMSmsMethod;
  /* 自定义ACTION常数,作为广播的Intent Filter识别常数 */
  public static String SMS_SEND_ACTIOIN = "SMS_SEND_ACTIOIN";
  public static String SMS_DELIVERED_ACTION = "SMS_DELIVERED_ACTION";

  /* 建立两个mServiceReceiver对象,作为类成员变量 */
  private SMSReceiver mSendSMSReceiver, mDeliveredSMSReceiver;
  
  private Context mContext;

  private SMSMethod(Context context){
    mCOntext=context;
    registerReceiver();

  }

  public static SMSMethod getInstance(Context context){
    if (mSMSmsMethod==null){
      synchronized (SMSMethod.class){
        if (mSMSmsMethod==null){
          mSMSmsMethod=new SMSMethod(context);
        }
      }
    }
    return mSMSmsMethod;
  }

  /**
   * 注册
   */
  public void registerReceiver(){
     /* 自定义IntentFilter为SENT_SMS_ACTIOIN Receiver */
    IntentFilter mFilter01;
    mFilter01 = new IntentFilter(SMS_SEND_ACTIOIN);
    mSendSMSReceiver = new SMSReceiver();
    mContext.registerReceiver(mSendSMSReceiver, mFilter01);

    /* 自定义IntentFilter为DELIVERED_SMS_ACTION Receiver */
    mFilter01 = new IntentFilter(SMS_DELIVERED_ACTION);
    mDeliveredSMSReceiver = new SMSReceiver();
    mContext.registerReceiver(mDeliveredSMSReceiver, mFilter01);
  }

  public void unregisterReceiver(){
    /* 取消注册自定义Receiver */
    if (mSendSMSReceiver!=null){
      mContext.unregisterReceiver(mSendSMSReceiver);
    }
    if (mDeliveredSMSReceiver!=null) {
      mContext.unregisterReceiver(mDeliveredSMSReceiver);
    }
  }

  public void SendMessage(String strDestAddress,String strMessage){
    /* 建立SmsManager对象 */
    SmsManager smsManager = SmsManager.getDefault();
    try {
     /* 建立自定义Action常数的Intent(给PendingIntent参数之用) */
      Intent itSend = new Intent(SMS_SEND_ACTIOIN);
      Intent itDeliver = new Intent(SMS_DELIVERED_ACTION);

     /* sentIntent参数为传送后接受的广播信息PendingIntent */
      PendingIntent mSendPI = PendingIntent.getBroadcast(mContext, 0, itSend, 0);

     /* deliveryIntent参数为送达后接受的广播信息PendingIntent */
      PendingIntent mDeliverPI = PendingIntent.getBroadcast(mContext, 0, itDeliver, 0);
      List divideCOntents= smsManager.divideMessage(strMessage);
      for (String text:divideContents) {
         /* 发送SMS短信,注意倒数的两个PendingIntent参数 */
        smsManager.sendTextMessage(strDestAddress, null, text, mSendPI, mDeliverPI);
      }

    }catch(Exception e) {
      e.printStackTrace();
    }
  }
  public void SendMessage2(String strDestAddress,String strMessage){
    ArrayList sentPendingIntents = new ArrayList();
    ArrayList deliveredPendingIntents = new ArrayList();


    /* 建立SmsManager对象 */
    SmsManager smsManager = SmsManager.getDefault();
    try {
     /* 建立自定义Action常数的Intent(给PendingIntent参数之用) */
      Intent itSend = new Intent(SMS_SEND_ACTIOIN);
      Intent itDeliver = new Intent(SMS_DELIVERED_ACTION);

     /* sentIntent参数为传送后接受的广播信息PendingIntent */
      PendingIntent mSendPI = PendingIntent.getBroadcast(mContext, 0, itSend, 0);

     /* deliveryIntent参数为送达后接受的广播信息PendingIntent */
      PendingIntent mDeliverPI = PendingIntent.getBroadcast(mContext, 0, itDeliver, 0);
      ArrayList mSMSMessage = smsManager.divideMessage(strMessage);

      for (int i = 0; i 

2、短信发送状态的监听

package com.javen.sms.receiver;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.telephony.SmsManager;
import android.widget.Toast;

/**
 * Created by Javen on 2016-03-15.
 */
public class SMSReceiver extends BroadcastReceiver{

  @Override
  public void onReceive(Context context, Intent intent) {
    if (intent.getAction().equals(SMSMethod.SMS_SEND_ACTIOIN)){
      try{
        /* android.content.BroadcastReceiver.getResultCode()方法 */
        //Retrieve the current result code, as set by the previous receiver.
        switch (getResultCode()){
          case Activity.RESULT_OK:
            System.out.println("短信发送成功");
            Toast.makeText(context, "短信发送成功", Toast.LENGTH_SHORT).show();
            break;
          case SmsManager.RESULT_ERROR_GENERIC_FAILURE:
            System.out.println("短信发送失败");
            Toast.makeText(context, "短信发送失败", Toast.LENGTH_SHORT).show();
            break;
          case SmsManager.RESULT_ERROR_RADIO_OFF:
            break;
          case SmsManager.RESULT_ERROR_NULL_PDU:
            break;
        }
      }catch (Exception e){
        e.printStackTrace();
      }
    } else if(intent.getAction().equals(SMSMethod.SMS_DELIVERED_ACTION)){
      /* android.content.BroadcastReceiver.getResultCode()方法 */
      switch(getResultCode()){
        case Activity.RESULT_OK:
          System.out.println("短信已送达");
          Toast.makeText(context, "短信已送达", Toast.LENGTH_SHORT).show();
          break;
        case SmsManager.RESULT_ERROR_GENERIC_FAILURE:
          System.out.println("短信未送达");
          /* 短信未送达 */
          Toast.makeText(context, "短信未送达", Toast.LENGTH_SHORT).show();
          break;
        case SmsManager.RESULT_ERROR_RADIO_OFF:
          break;
        case SmsManager.RESULT_ERROR_NULL_PDU:
          break;
      }
    }
  }
}

测试代码:

public void sendTextMessage(View view){
    SMSMethod.getInstance(this).SendMessage("xxxx","测试短信。。。");
  }
  public void sendMultipartTextMessage(View view){
    SMSMethod.getInstance(this).SendMessage2("xxxx", "测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。测试短信wwww。。。");
  }

  @Override
  protected void onPause() {
    SMSMethod.getInstance(this).unregisterReceiver();
    super.onPause();
  }

别忘了权限的问题:

 

 以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • 【Windows】实现微信双开或多开的方法及步骤详解
    本文介绍了在Windows系统下实现微信双开或多开的方法,通过安装微信电脑版、复制微信程序启动路径、修改文本文件为bat文件等步骤,实现同时登录两个或多个微信的效果。相比于使用虚拟机的方法,本方法更简单易行,适用于任何电脑,并且不会消耗过多系统资源。详细步骤和原理解释请参考本文内容。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文讲述了作者通过点火测试男友的性格和承受能力,以考验婚姻问题。作者故意不安慰男友并再次点火,观察他的反应。这个行为是善意的玩人,旨在了解男友的性格和避免婚姻问题。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
author-avatar
陈宏儒64721
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有