手机短信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; } } }