程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> PHP通過串口實現發送短信

PHP通過串口實現發送短信

編輯: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; 
} 
  
?> 

以上所述就是本文的全部內容了,希望大家能夠喜歡。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved