手机短信PDU编码与解码



 编码代码:

/////////////////////////////////////
///文    件:PDUdecoding.cs
///程 序 员:klude
///编制日期:2007-11-04
///概    要:针对国内短信编码(USC2)
///组成结构:包含四个函数:
///          smsDecodedCenterNumber(string srvCenterNumber)            短信中心号编码
///          smsPDUEncoded(string srvContent)                          短信内容编码
///          smsDecodedNumber(string srvNumber)                        接收短信手机号编码
///          smsDecodedsms(string strCenterNumber, string strNumber, string strSMScontent)   整个短信的编码
///          一个字段
///          string nLength;   //要发送内容的长度,由两部分组成,接收手机号加上要发送的内容
///          
///版    本:1.0.20071104
///用    法:
///         1,把文件拷贝到你的项目中,添加引用  using SMS;
///         2,定义本类,例如:PDUdecoding ss = new SMS.PDUdecoding();
///         3,现在你就可以使用本类了,例如:
///         textBox2.Text = ss.smsDecodedsms(txtCenterNumber.Text,txtNumber.Text,textBox1.Text);

////////////////////////////////////
using System;
using System.Text;

namespace SMS
{
	public class PDUdecoding
	{
		
		public string nLength;   //要发送内容的长度,由两部分组成,接收手机号加上要发送的内容
		/// <summary>
			/// 函数功能:短信内容编码
			/// 函数名称:smsPDUEncoded(string srvContent)
			/// 参    数:srvContent 要进行转换的短信内容,string类型
			/// 返 回 值:编码后的短信内容,string类型
			/// 程 序 员:klude
            /// 编制日期:2007-11-04
			/// 函数说明:
			///          1,采用Big-Endian 字节顺序的 Unicode 格式编码,也就说把高低位的互换在这里完成了
			///          2,将转换后的短信内容存进字节数组
			///          3,去掉在进行Unicode格式编码中,两个字节中的"-",例如:00-21,变成0021
			///          4,将整条短信内容的长度除2,保留两位16进制数
			/// </summary>
		public string smsPDUEncoded(string srvContent)
			{
				Encoding encodingUTF = System.Text.Encoding.BigEndianUnicode;
				string s = null;
				byte [] encodedBytes = encodingUTF.GetBytes(srvContent);
				for (int i =0;i < encodedBytes.Length;i++)
				{
					s += BitConverter.ToString(encodedBytes,i,1);
				}
				s = String.Format("{0:X2}{1}",s.Length / 2,s);
				
				return s;
			}

		/// <summary>
		/// 函数功能:短信中心号编码
		/// 函数名称:smsDecodedCenterNumber(string srvCenterNumber)
		/// 参    数:srvCenterNumber 要进行转换的短信中心号,string类型
		/// 返 回 值:编码后的短信中心号,string类型
		/// 程 序 员:klude
        /// 编制日期:2007-11-04
		/// 函数说明:
		///          1,将奇数位和偶数位交换。
		///          2,短信中心号奇偶数交换后,看看长度是否为偶数,如果不是,最后添加F
		///          3,加上短信中心号类型,91为国际化
		///          4,计算编码后的短信中心号长度,并格化成二位的十六进制
		/// </summary>
		public string smsDecodedCenterNumber(string srvCenterNumber)
		{
			string s = null;
            if (!(srvCenterNumber.Substring(0, 2) == "86"))
            {
                srvCenterNumber = String.Format("86{0}", srvCenterNumber);     //检查当前短信中心号是否按标准格式书写,不是,就补上“86”
            }
			int nLength = srvCenterNumber.Length;
			for(int i = 1 ; i < nLength;i += 2)                       //奇偶互换
			{
				s += srvCenterNumber[i];
				s += srvCenterNumber[i-1];
			}
			if(!(nLength % 2 == 0))                           //是否为偶数,不是就加上F,并对最后一位与加上的F位互换
			{
				s += 'F';
				s += srvCenterNumber[nLength - 1];
			}
			s = String.Format("91{0}",s);                    //加上91,代表短信中心类型为国际化
			s = String.Format("{0:X2}{1}",s.Length / 2,s);   //编码后短信中心号长度,并格式化成二位十六制
			return s;
		}

		/// <summary>
		/// 函数功能:接收短信手机号编码
		/// 函数名称:smsDecodedNumber(string srvNumber)
		/// 参    数:srvCenterNumber 要进行转换的短信中心号,string类型
		/// 返 回 值:编码后的接收短信手机号,string类型
		/// 程 序 员:klude
		/// 编制日期:2007-11-04
		/// 函数说明:
		///          1,检查当前接收手机号是否按标准格式书写,不是,就补上“86”
		///          1,将奇数位和偶数位交换。
		///          2,短信中心号奇偶数交换后,看看长度是否为偶数,如果不是,最后添加F
		/// </summary>
		public string smsDecodedNumber(string srvNumber)
		{
			string s = null;
            if (!(srvNumber.Substring(0, 2) == "86"))
            {
                srvNumber = String.Format("86{0}", srvNumber);     //检查当前接收手机号是否按标准格式书写,不是,就补上“86”
            }
			int nLength = srvNumber.Length;
			for(int i = 1 ; i < nLength ; i += 2)                 //将奇数位和偶数位交换
			{
				s += srvNumber[i];
				s += srvNumber[i-1];
			}
			if(!(nLength % 2 == 0))                              //是否为偶数,不是就加上F,并对最后一位与加上的F位互换
			{
				s += 'F';
				s += srvNumber[nLength - 1];
			}
			return s;
		}

		/// <summary>
		/// 函数功能:整个短信的编码
		/// 函数名称:smsDecodedsms(string strCenterNumber, string strNumber, string strSMScontent)
		/// 参    数:strCenterNumber 要进行转换的短信中心号,string类型
		///           strNumber       接收手机号码,string类型
		///           strSMScontent   短信内容
		/// 返 回 值:完整的短信编码,可以在AT指令中执行,string类型
		/// 程 序 员:klude
        /// 编制日期:2007-11-04
		/// 函数说明:
		///           11000D91和000800   在国内,根据PDU编码原则,我们写死在此,详细解释请看我的文章    
        ///           31000D91//短信报告
		/// </summary>
		public string smsDecodedsms(string strCenterNumber, string strNumber, string strSMScontent)
		{
			string s = String.Format("{0}11000D91{1}000800{2}",smsDecodedCenterNumber(strCenterNumber),smsDecodedNumber(strNumber),smsPDUEncoded(strSMScontent));
			nLength =String.Format("{0:D2}", (s.Length - smsDecodedCenterNumber(strCenterNumber).Length) / 2 );   //获取短信内容加上手机号码长度
			return s;
		}
	}
}


//sms的pdu编码规则 
 
//目前,发送短消息常用Text和PDU(Protocol Data Unit,协议数据单元)模式。使用Text模式收发短信代码简单,实现起来十分容易,但最大的缺点是不能收发中文短信;而PDU模式不仅支持中文短信,也能发送英文短信。PDU模式收发短信可以使用3种编码:7-bit、8-bit和UCS2编码。7-bit编码用于发送普通的ASCII字符,8-bit编码通常用于发送数据消息,UCS2编码用于发送Unicode字符。一般的PDU编码由A B C D E F G H I J K L M十三项组成。 
//A:短信息中心地址长度,2位十六进制数(1字节)。
//B:短信息中心号码类型,2位十六进制数。
//C:短信息中心号码,B+C的长度将由A中的数据决定。
//D:文件头字节,2位十六进制数。 
//E:信息类型,2位十六进制数。
//F:被叫号码长度,2位十六进制数。
//G:被叫号码类型,2位十六进制数,取值同B。
//H:被叫号码,长度由F中的数据决定。
//I:协议标识,2位十六进制数。
//J:数据编码方案,2位十六进制数。
//K:有效期,2位十六进制数。
//L:用户数据长度,2位十六进制数。
//M:用户数据,其长度由L中的数据决定。J中设定采用UCS2编码,这里是中英文的Unicode字符。

//PDU编码协议简单说明

//例1 发送:SMSC号码是+8613800250500,对方号码是13693092030,消息内容是“Hello!”。从手机发出的PDU串可以是
//08 91 68 31 08 20 05 05 F0 11 00 0D 91 68 31 96 03 29 30 F0 00 00 00 06 C8 32 9B FD 0E 01
//对照规范,具体分析:
//分段 含义 说明
//08 SMSC地址信息的长度 共8个八位字节(包括91)
//91 SMSC地址格式(TON/NPI) 用国际格式号码(在前面加‘+’)
//68 31 08 20 05 05 F0 SMSC地址 8613800250500,补‘F’凑成偶数个
//11 基本参数(TP-MTI/VFP) 发送,TP-VP用相对格式
//00 消息基准值(TP-MR) 0
//0D 目标地址数字个数 共13个十进制数(不包括91和‘F’)
//91 目标地址格式(TON/NPI) 用国际格式号码(在前面加‘+’)
//68 31 96 03 29 30 F0 目标地址(TP-DA) 8613693092030,补‘F’凑成偶数个
//00 协议标识(TP-PID) 是普通GSM类型,点到点方式
//00 用户信息编码方式(TP-DCS) 7-bit编码
//00 有效期(TP-VP) 5分钟
//06 用户信息长度(TP-UDL) 实际长度6个字节
//C8 32 9B FD 0E 01 用户信息(TP-UD) “Hello!”

//例2 接收:SMSC号码是+8613800250500,对方号码是13693092030,消息内容是“你好!”。手机接收到的PDU串可以是
//08 91 68 31 08 20 05 05 F0 84 0D 91 68 31 96 03 29 30 F0 00 08 30 30 21 80 63 54 80 06 4F 60 59 7D 00 21
//对照规范,具体分析:
//分段 含义 说明
//08 地址信息的长度 个八位字节(包括91)
//91 SMSC地址格式(TON/NPI) 用国际格式号码(在前面加‘+’)
//68 31 08 20 05 05 F0 SMSC地址 8613800250500,补‘F’凑成偶数个
//84 基本参数(TP-MTI/MMS/RP) 接收,无更多消息,有回复地址
//0D 回复地址数字个数 共13个十进制数(不包括91和‘F’)
//91 回复地址格式(TON/NPI) 用国际格式号码(在前面加‘+’)
//68 31 96 03 29 30 F0 回复地址(TP-RA) 8613693092030,补‘F’凑成偶数个
//00 协议标识(TP-PID) 是普通GSM类型,点到点方式
//08 用户信息编码方式(TP-DCS) UCS2编码
//30 30 21 80 63 54 80 时间戳(TP-SCTS) 2003-3-12 08:36:45  +8时区
//06 用户信息长度(TP-UDL) 实际长度6个字节
//4F 60 59 7D 00 21 用户信息(TP-UD) “你好!”


//若基本参数的最高位(TP-RP)为0,则没有回复地址的三个段。从Internet上发出的短消息常常是这种情形。
//注意号码和时间的表示方法,不是按正常顺序顺着来的,而且要以‘F’将奇数补成偶数。


//在PDU Mode中,可以采用三种编码方式来对发送的内容进行编码,它们是7-bit、8-bit和UCS2编码。7-bit编码用于发送普通的ASCII字符,它将一串7-bit的字符(最高位为0)编码成8-bit的数据,每8个字符可“压缩”成7个;8-bit编码通常用于发送数据消息,比如图片和铃声等;而UCS2编码用于发送Unicode字符。PDU串的用户信息(TP-UD)段最大容量是140字节,所以在这三种编码方式下,可以发送的短消息的最大字符数分别是160、140和70。这里,将一个英文字母、一个汉字和一个数据字节都视为一个字符。


//需要注意的是,PDU串的用户信息长度(TP-UDL),在各种编码方式下意义有所不同。7-bit编码时,指原始短消息的字符个数,而不是编码后的字节数。8-bit编码时,就是字节数。UCS2编码时,也是字节数,等于原始短消息的字符数的两倍。如果用户信息(TP-UD)中存在一个头(基本参数的TP-UDHI为1),在所有编码方式下,用户信息长度(TP-UDL)都等于头长度与编码后字节数之和。如果采用GSM 03.42所建议的压缩算法(TP-DCS的高3位为001),则该长度也是压缩编码后字节数或头长度与压缩编码后字节数之和。
 



解码代码:

/////////////////////////////////////   
///文         件:FFPDUdecoding.cs   
///概         要:针对国内短信解码(USC2)   
///组成结构:包含四个函数:   
///       1、GetEverySMS(string   SMS)   
///       2、GetTelphone(string   SMS)   
///       3、GetDataTime(string   SMS)     
///       4、GetContent(string     SMS)         
////////////////////////////////////   
using System;
using System.Text;
namespace SMS
{
    ///   <summary>   
    ///   FPDUdecoding   的摘要说明。   
    ///   </summary>   
    public class FPDUdecoding
    {
        public FPDUdecoding()
        {

            //   TODO:   在此处添加构造函数逻辑   

        }

        ///   <summary>   
        ///   判断接受的短信是PDU格式还是TEXT格式   
        ///   </summary>   

        public bool IsPDU(string SMS)
        {
            if (SMS.Substring(40, 2) != "08")
                return false;
            return true;
        }

        ///   <summary>   
        ///   函数功能:短信内容提取   
        ///   函数名称:GetEverySMS(string   SMS)   
        ///   参         数:SMS   要进行提取的整个短信内容   
        ///   返   回   值:将多个短信内容拆分   
        ///   </summary>   

        public string[] GetEverySMS(string SMS)
        {
            char[] str = "\n".ToCharArray();
            string[] temp = SMS.Split(str);
            return temp;
        }

        ///   <summary>   
        ///   函数功能:提取短信的发送人电话号码   
        ///   函数名称:GetTelphone(string   SMS)   
        ///   参         数:SMS   要进行转换的整个短信内容   
        ///   返   回   值:电话号码   
        ///   </summary>   

        public string GetTelphone(string SMS)
        {
            string tel = SMS.Substring(26, 12);
            string s = "";
            for (int i = 0; i < 9; i += 2)
            {
                s += tel[i + 1];
                s += tel[i];
            }
            s += tel[tel.Length - 1];
            return s;
        }

        ///   <summary>   
        ///   函数功能:提取短信的发送时间   
        ///   函数名称:GetDataTime(string   SMS)   
        ///   参         数:SMS:要进行转换的整个短信内容   
        ///   返   回   值:发送时间   
        ///   </summary>   

        public string GetDataTime(string SMS)
        {
            string time = SMS.Substring(42, 12);
            string s = "";
            for (int i = 0; i < 11; i += 2)
            {
                s += time[i + 1];
                s += time[i];
            }
            string t = s.Substring(0, 2) + "年" + s.Substring(2, 2) + "月" + s.Substring(4, 2) + "日" + s.Substring(6, 2) + ":" + s.Substring(8, 2) + ":" + s.Substring(10, 2);
            return t;
        }

        ///   <summary>   
        ///   函数功能:提取短信的内容(PDU)   
        ///   函数名称:GetContent(string   SMS)   
        ///   参         数:SMS:要进行转换的整个短信内容   
        ///   返   回   值:短信内容   
        ///   </summary>   

        public string GetContent(string SMS)
        {
            string c = "";
            string len = SMS.Substring(56, 2);
            int length = System.Convert.ToInt16(len, 16);
            length *= 2;
            string content = SMS.Substring(58, length);
            for (int i = 0; i < length; i += 4)
            {
                string temp = content.Substring(i, 4);
                int by = System.Convert.ToInt16(temp, 16);
                char ascii = (char)by;
                c += ascii.ToString();
            }
            return c;
        }

        ///   <summary>   
        ///   函数功能:提取短信的TEXT内容(TEXT)   
        ///   函数名称:GetTextContent(string   SMS)   
        ///   参         数:SMS:要进行转换的整个短信内容   
        ///   返   回   值:短信内容   
        ///   </summary>   

        public string GetTextContent(string SMS)
        {
            string str = "";
            string c = "";
            byte by;
            char ascii;
            int i;
            SMS = SMS.Replace("\r", "");
            SMS = SMS.Replace("\n", "");
            string content = SMS.Substring(58);
            for (i = content.Length - 2; i >= 0; i -= 2)
            {
                by = Convert.ToByte(content.Substring(i, 2), 16);
                str += Convert.ToString(by, 2).PadLeft(8, '0');
            }
            for (i = str.Length - 7; i >= 0; i -= 7)
            {
                by = Convert.ToByte(str.Substring(i, 7), 2);
                ascii = (char)by;
                c += ascii.ToString();
            }
            return c;
        }
    }
}



posted @ 2012-03-20 14:42  java简单例子  阅读(1234)  评论(1编辑  收藏  举报