PHP通过串口实现发送短信
随技术进步,短信收发领域按时间先后产生了三种模式: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; } ?>
以上所述就是本文的全部内容了,希望大家能够喜欢。