PHP通过串口实现发送短信


Posted in PHP onJuly 08, 2015

随技术进步,短信收发领域按时间先后产生了三种模式:BLOCK MODE,基于AT指令的TEXT MODE,基于AT指令的PDU MODE。其中,TEXT MODE比较简单,多款诺基亚手机均支持此款模式。西门子的手机大多数只支持PDU MODE。PDU 模式是收发短信的一种方法,短信正文经过十六进制编码后被传送。目前,PDU已取代BLOCK MODE。

SMS是由Etsi所制定的一个规范(GSM 03.40 和GSM 03.38)。当使用7-bits编码时,它可以发送最多160个字符;但用8-bit编码,最多可以发送140个字符,通常无法直接通过手机显示;还有用16-bit编码时,最多70个字符,被用来显示Unicode(UCS2)文本信息,可以被大多数的手机所显示。

     今天讨论的是PDU MODE,UCS2编码,也就是说,最多只能发送70个字符,不管英文还是中文。
     假设现在要发送如下信息:“你好”。在没有发送之前,要知道手机SIM卡所在地的短信中心号,例如移动的短信中心号:

      接收的手机号:13638197275
      杭州短信中心号:13800571500
      短信内容: 你好
    发送这条短信,要进行编码后手机才会执行,编码后会变成以下一串字符:
0891683180501705F011000D91683136187972F5000800044F60597D
   看不懂吧,从头到尾把这串编码解释一下:
       08 ? 指的是短信中心号的长度,也就是指(91)+(683180501705F0)的长度除以2,即 08 =(2+14)/ 2
       91 ? 指的是短信息中心号码类型。91是TON/NPI遵守International/E.164标准,指在号码前需加‘+'号;此外还有其它数值,但91最常用。
       683180501705F0  - 短信息中心号码。由于位置上略有处理,实际号码应为:8613800571500(字母F是补足偶数长度添加的字符)。
       11 - 文件头字节
       00 - 信息类型(TP-Message-Reference)
       0D - 被叫号码长度
       91 - 被叫号码类型

其实在实际处理中,我们通常把11000D91写死在程序中,因为在国内,这些数据都是不会改变的。

       683136187972F5 - 被叫号码,经过了位移处理,实际号码为“8613638197275”。

    上面的(00 )+(0D )+(91 )+(683136187972F5 ),构成了整个短信的第二部份目的地址(TP-Destination-Address)。

    继续...
    00 - 协议标识TP-PID,这里一般为00
    08 - 数据编码方案TP-DCS(TP-Data-Coding-Scheme),采用前面说的USC2(16bit)数据编码
    00 - 有效期TP-VP(TP-Valid-Period)
    04  - 长度TP-UDL(TP-User-Data-Length),也就是信息长度/2的十六进04
    4F60597D 这里就是短信内容了,实际内容为:“你好”

   根据以上情况,就可以写出短信编码的程序脚本了。

一、短信中心号码处理:

1、将短信息中心号码“+8613800571500”去掉+号,看长度是否为偶数,如果不是,最后添加F
=> “8613800571500F”
2、将奇数位和偶数位交换。
=> “683108501705F0″
3、将短信息中心号码前面加上字符91,91是国际化的意思
=> “91683108501705F0″
4、算出长度,结果除2,格式化成2位的16进制字符串,16 / 2 = 8 => “08″
=> “0891683108501705F0″

二、手机号码处理:

1、将手机号码+8613638197275去掉+号,看看长度是否为偶数,如果不是,最后添加F
=> “8613638197275F”
2、将手机号码奇数位和偶数位交换。
=> “683136187972F5″

三、短信息部分处理:

1、转字符串转换为Unicode代码,
“你好”的unicode代码 为4F60597D
2、将长度除2,保留两位16进制数,即 4F60597D = 8 / 2 => “04″,
=> “044F60597D″

四、组合

1、手机号码前加上字符串 11000D91(1100:固定,0D:手机号码的长度,不算+号,十六进制表示,91:发送
到手机为91,发送到小灵通为81),
即 11000D91 + 683136187972F5
=> 11000D91683136187972F5
2、手机号码后加上 000800 和刚才的短信息内容,000800也写死就可以了
即 11000D91683136187972F5 + 000800 + 044F60597D
=>  11000D91683136187972F5000800044F60597D
3、整条信息长度除以2,格式化成2位的十进制数
即 11000D91683136187972F5000800044F60597D => 38位 / 2 => 19

五、所以要发送的内容为

AT+CMGF=0 <回车> #此处为设定短信发送模式PDU
OK
AT+CMGS=19<回车>
> #输入短信内容编码

附加最终PHP代码:

<?php 
// Requirement dio, use cmd install: pecl install dio 
set_time_limit(0); 
  
// Windows use COM1: 
$fd=dio_open('/dev/ttyS0', O_RDWR); 
if(!$fd) 
{ 
  die("打开串口ttyS0失败"); 
} 
  
// dio_tcsetattr() only Linux 
// Windows 使用 exec('mode COM1: baud=9600 data=8 stop=1 parity=n xon=on'); 
dio_tcsetattr($fd, array( 
 'baud' => 9600, 
 'bits' => 8, 
 'stop' => 1, 
 'parity' => 0 
)); 
  
//$ff=dio_stat($fd); 
//print_r($ff); 
//echo "GSM AT is start on ttyS0\n"; 
  
//短信中心号码 
$smsc = "8613800571500"; 
$invert_smsc = invertNumbers($smsc); // 转换短信中心号码 
$inter = chr(13); // 回车字符 
  
$ctrlz = chr(26); // ctrl+z 
  
// 发送信息 
$text 
  = '你好'; 
$send_to = '8613638197275'; 
$pdu_phone = hex2str(utf82unicode($text)); 
$pdu_phone = sprintf("%02X", strlen($pdu_phone)/2) . $pdu_phone; 
$pdu_phone = '11000D91' . invertNumbers($send_to) . '000800' . $pdu_phone; 
$atcmd   = 'AT+CMGF=0' . $inter; 
@dio_write($fd, $atcmd); 
$atcmd   = 'AT+CMGS=' . sprintf("%d", strlen($pdu_phone)/2) . $inter; 
@dio_write($fd, $atcmd); 
$pdu_addr  = '0891' . invertNumbers($smsc); 
$pdu_all  = $pdu_addr . $pdu_phone . $ctrlz . $inter; 
@dio_write($fd, $pdu_all); 
dio_close($fd); 
  
// 我的是utf-8编码 
function utf82unicode($str)  
{ 
  return iconv("utf-8", "UCS-2BE", $str); 
} 
  
function hex2str($hexstring)  
{ 
  $str = ''; 
  for($i = 0, $len = strlen($hexstring); $i < $len; $i++) 
  { 
    $str .= sprintf("%02X", ord(substr($hexstring, $i, 1))); 
  } 
  return $str; 
} 
  
function invertNumbers($msisdn)  
{ 
  $len = strlen($msisdn); 
  if ( 0 != fmod($len, 2) ) 
  { 
    $msisdn .= "F"; 
    $len = $len + 1; 
  } 
  
  for ($i=0; $i<$len; $i+=2) 
  { 
    $t = $msisdn[$i]; 
    $msisdn[$i] = $msisdn[$i+1]; 
    $msisdn[$i+1] = $t; 
  } 
  return $msisdn; 
} 
  
?>

以上所述就是本文的全部内容了,希望大家能够喜欢。

PHP 相关文章推荐
支持oicq头像的留言簿(二)
Oct 09 PHP
PHP新手上路(十三)
Oct 09 PHP
php 购物车的例子
May 04 PHP
php类中private属性继承问题分析
Nov 01 PHP
php抓取页面的几种方法详解
Jun 17 PHP
PHP缓存机制Output Control详解
Jul 14 PHP
Yii扩展组件编写方法实例分析
Jun 29 PHP
php简单判断两个字符串是否相等的方法
Jul 13 PHP
浅谈PHP值mysql操作类
Jun 29 PHP
详解PHP安装mysql.so扩展的方法
Dec 31 PHP
PHP实现分布式memcache设置web集群session同步的方法
Apr 10 PHP
分析php://output和php://stdout的区别
May 06 PHP
PHP使用CURL模拟登录的方法
Jul 08 #PHP
一个PHP实现的轻量级简单爬虫
Jul 08 #PHP
PHP设置进度条的方法
Jul 08 #PHP
smarty内部日期函数html_select_date()用法实例分析
Jul 08 #PHP
php类的定义与继承用法实例
Jul 07 #PHP
php抽象类用法实例分析
Jul 07 #PHP
PHP导入导出Excel代码
Jul 07 #PHP
You might like
收音机另类DIY - 纸巾盒做外壳
2021/03/02 无线电
法国:浪漫之都的咖啡文化
2021/03/03 咖啡文化
php通过array_push()函数添加多个变量到数组末尾的方法
2015/03/18 PHP
gearman中worker常驻后台,导致MySQL server has gone away的解决方法
2020/02/27 PHP
基于jQuery的投票系统显示结果插件
2011/08/12 Javascript
javascript使用prototype完成单继承
2014/12/24 Javascript
javascript获取重复次数最多的字符
2015/07/08 Javascript
js+css实现上下翻页相册代码分享
2015/08/18 Javascript
JS实现网页Div层Clone拖拽效果
2015/09/26 Javascript
JS中input表单隐藏域及其使用方法
2017/02/13 Javascript
jQuery 禁止表单用户名、密码自动填充功能
2017/10/30 jQuery
使用Vue+Django+Ant Design做一个留言评论模块的示例代码
2020/06/01 Javascript
详解JavaScript执行模型
2020/11/16 Javascript
动态实现element ui的el-table某列数据不同样式的示例
2021/01/22 Javascript
[03:48]大碗DOTA
2019/07/25 DOTA
[52:06]完美世界DOTA2联赛决赛日 Inki vs LBZS 第一场 11.08
2020/11/10 DOTA
python 性能优化方法小结
2017/03/31 Python
Python 实现链表实例代码
2017/04/07 Python
Python3.5 创建文件的简单实例
2018/04/26 Python
python 美化输出信息的实例
2018/10/15 Python
python+pyqt5编写md5生成器
2019/03/18 Python
Python pip替换为阿里源的方法步骤
2019/07/02 Python
python错误调试及单元文档测试过程解析
2019/12/19 Python
如何通过python实现全排列
2020/02/11 Python
易程科技软件测试笔试
2013/03/24 面试题
九年级家长会邀请函
2014/01/15 职场文书
体育专业学生自我评价范文
2014/01/17 职场文书
九年级化学教学反思
2014/01/28 职场文书
公司司机岗位职责范本
2014/03/03 职场文书
六五普法规划实施方案
2014/03/21 职场文书
建设投标担保书
2014/05/13 职场文书
关于旅游的活动方案
2014/08/15 职场文书
2014年行政助理工作总结
2014/11/19 职场文书
2015元旦家电促销活动策划方案
2014/12/09 职场文书
2015年转正工作总结范文
2015/04/02 职场文书
机械原理课程设计心得体会
2016/01/15 职场文书