PHP通过串口实现发送短信
随技术进步,短信收发领域按时间先后产生了三种模式:BLOCKMODE,基于AT指令的TEXTMODE,基于AT指令的PDUMODE。其中,TEXTMODE比较简单,多款诺基亚手机均支持此款模式。西门子的手机大多数只支持PDUMODE。PDU模式是收发短信的一种方法,短信正文经过十六进制编码后被传送。目前,PDU已取代BLOCKMODE。
SMS是由Etsi所制定的一个规范(GSM03.40和GSM03.38)。当使用7-bits编码时,它可以发送最多160个字符;但用8-bit编码,最多可以发送140个字符,通常无法直接通过手机显示;还有用16-bit编码时,最多70个字符,被用来显示Unicode(UCS2)文本信息,可以被大多数的手机所显示。
今天讨论的是PDUMODE,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
//Requirementdio,usecmdinstall:peclinstalldio
set_time_limit(0);
//WindowsuseCOM1:
$fd=dio_open('/dev/ttyS0',O_RDWR);
if(!$fd)
{
die("打开串口ttyS0失败");
}
//dio_tcsetattr()onlyLinux
//Windows使用exec('modeCOM1:baud=9600data=8stop=1parity=nxon=on');
dio_tcsetattr($fd,array(
'baud'=>9600,
'bits'=>8,
'stop'=>1,
'parity'=>0
));
//$ff=dio_stat($fd);
//print_r($ff);
//echo"GSMATisstartonttyS0\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编码
functionutf82unicode($str)
{
returniconv("utf-8","UCS-2BE",$str);
}
functionhex2str($hexstring)
{
$str='';
for($i=0,$len=strlen($hexstring);$i<$len;$i++)
{
$str.=sprintf("%02X",ord(substr($hexstring,$i,1)));
}
return$str;
}
functioninvertNumbers($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;
}
?>
以上所述就是本文的全部内容了,希望大家能够喜欢。