** (十一) ** ** 短信部分—— VB.NET ** ** 编码 PDU **
PDU 的编码器的工作原理是解码器的逆过程。根据需要编码器只需要编码发送的 PDU 代码,工作相对简单。本文讲解编码思路,具体代码请参考 Blog 中 PDUEncoder 部分
我把 PDU 的编码分为两部分, SMS 和 EMS 。 EMS 部分我只提供了 ConcatenatedShortMessage 的编码器。这是超长短信的编码,用得最多。
** SMS ** ** 编码 **
编码一个 SMS 一般需要如下的信息:
TP_Data_Coding_Scheme TP_UD 编码方式
TP_Destination_Address 对方号码
TP_Message_Reference 参考号码
TP_Status_Report_Request 状态报告
TP_User_Data 用户信息
TP_Validity_Priod 有效期
ServiceCenterNumber 短信中心号码
所以在编码器中存在以上的属性,并在 Set 中加入了处理代码,将可读信息转换成对应的十六进制信息。
特别注意的是 TP_User_Data 属性,它可以根据用户数据编码自动设置 TP_UDL 。对于纯英文编码, TP_UDL 为所有的字符数;对于 Unicode 编码,由于一个字符由两个字节表示, TP_UDL 为所有的字符数 2 。注意检查 TP_User_Data 的长度,对于 SMS 来说编码后的 TP_UD 长度不能超过 140 字节。也就是说英文 160 个字符( 140/78 ),中文 70 个字符。
对于 TP_UD 的编码在解码器中也有说明,在此不再赘述。
我还设计了几个枚举变量:
ENUM_TP_DCS 编码方式
ENUM_TP_SRI 状态报告
ENUM_TP_VALID_PERIOD 有效期
ENUM_TP_VPF 有效期格式
这些枚举变量可以简化输入,也利于日后扩充。
当以上内容设置好以后,基本上一个短信的架子就出来了。此时调用 GetSMSPDUCode 进行组合,简单的把十六进制拼接起来就形成了一个完整的 PDU 代码。
** EMS ** ** —— ConcatenatedShortMessage ** ** 部分 **
编码 EMS 较 SMS 复杂,但每条 EMS 的基础还是 SMS ,所以我直接继承了 SMS 类。区别主要是要处理好 TP_UD 和 IE 。对于 ConcatenatedShortMessage ,由于其 IE 和 TP_UDHL 占据了 TP_UD 的部分空间,所以每条短信英文只能容纳 133 字符,中文 66 字符。我们可以通过此信息得到短信条数。
如果 TP_DCS 为 Unicode 编码,则短信条目为:
TotalMessages = (TP_UD.Length / 4) \ 66 + ((TP_UD.Length / 4 Mod 66) = 0)+1
如果为 7bit ,则为:
TotalMessages = (tp_ud.Length \ 266) - ((tp_ud.Length Mod 266) = 0)+1
注意在程序中我为了简化以后的数组操作,就没有加一。
确定了短信条数以后通过一个循环就可以提取出每条短信的 TP_UD 。
Select Case tp_dcs
Case ENUM_TP_DCS.UCS2
tmpTP_UD = Mid(TP_UD, i * 66 * 4 + 1, 66 * 4) 'When TP_UDL is odd, the max length of an Unicode string in PDU code is 66 Charactor.See [3GPP TS 23.040 V 6.5.0 (2004-09] 9.2.3.24.1
Case ENUM_TP_DCS.DefaultAlphabet
tmpTP_UD = Mid(tp_ud, i * 133 * 2 + 1, 133 * 2)
End Select
此后还需要编码 IE 部分,关键代码是确定 TP_UDL 的值。对于 TP_DCS 为 7bit 来说确定此值显得比较复杂,弄不好容易出现多一个少一个的错误。
If tp_dcs = ENUM_TP_DCS.UCS2 Then
TP_UDL = tmpTP_UD.Length / 2 + 6 + 1 '6: length of IE
End If
If tp_dcs = ENUM_TP_DCS.DefaultAlphabet Then
TP_UDL = Fix((tmpTP_UD.Length + 7 * 2) * 4 / 7) '6:length of IE
End If
然后根据 3GPP 里关于 EMS 的结构的说明就可以编写出 EMS PDU 的处理程序。详见原代码。
如果需要扩展 EMS 以适应更多种类的 EMS ,可以参考 3GPP 写出更为强大的编码程序。但最关键的还是需要处理好 IE 以及 TP_UDL 。