隨技術進步,短信收發領域按時間先後產生了三種模式: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; } ?>
以上所述就是本文的全部內容了,希望大家能夠喜歡。