由于短信收到设备中,设备读取短信是以PDU格式显示的,但是咱们程序就需要对这类格式进行解释了,网上找了很多资料,但是都是不全,东一块西一块的。
首先来看一下我做的一些测试吧,看demo来讲解
// 07 683108501415F4 【1-12】服务中心的电话号码 // 00 【1】协议数据单元类型 // 05 {长度} A1 {类型} 0180F6 {地址}【2-12】发送方SME的地址 // 00 【1】参数显示SMSC以何种方式处理SM (比如FAX,、Voice等) // 08 【1】参数表示用户数据(UD)采用什么编码方案 // 11509051208023 【7】参数表示SMSC接收到消息时的时间戳 // 80 【1】用户数据段长度 // 5BF94E0D8D77FF0C60A853D19001768463074EE467098BEF002C8BF768385B9E540E91CD65B053D19001002E7F168F91002200310030003000380036002253D190015230003100300030003800364F7F752877ED4FE184254E1A5385002C67E58BE253CA529E74064E1A52A166F465B94FBF002EFF08653653D1514D8D39FF09 // 【0-140】SM数据 // ========================================================================= // 07 683108100005F0 // 60 // 10 A1 2125------------ //这里为发送者手机号,我给 - 替换了 // 00 // 08 // 11 50 91 70 34 20 23 // 8C // [05 00 03 DC 02 01] //<----这个为多条短信时,网上的其它文档没有说明 // 9EC46E90003A55114E2D4E2D4E2D867D8FDB6821594B8FDB00205730753C593A5730753C9ED166970020662F4E2D4E2D4E2D4E2D4E2D67AF5706660E56ED593A5730753C534F67E58F698FDB534F67E557288FDB56E065F6534F56FD5730753C534F67E5662F4E2D4E2D4E2D4E2D4E2D67AF5706660E56ED593A5730753C534F67E58F698FDB
我的思路是把上面的信息一块块读取出来。。所以方法比较笨。
1. 取短信中心号长度
就是上面的最开始2位,如:07
static String getServiceCenter(SmsBean smsBean , String sms) { // 如 : 07 * 2 + 2就得到短信中心号的所有长度 smsBean.setServiceCenterLength(smsBean.getServiceCenterLength() * 2 + 2); String centerNumber = sms.substring(0, smsBean.getServiceCenterLength()); smsBean.setPduCenterNumber(centerNumber); log.debug("PDU格式短信中心号码:" + smsBean.getPduCenterNumber()); smsBean.setTemp(sms.replaceAll(smsBean.getPduCenterNumber(), "").substring(2)); centerNumber = centerNumber.substring(2); StringBuilder builder = new StringBuilder(); // 把号码调整后去除86开头或者由于不足偶数引起补全的F字符 for (int i = 0; i < centerNumber.length(); i += 2) { builder.append(centerNumber.charAt(i + 1)).append(centerNumber.charAt(i)); } return builder.toString().replaceAll("^86|F", ""); }
2.提示发件人号码
static String getTelphone(SmsBean smsBean) { // 读取长度 int beginIndex = Integer.parseInt(smsBean.getTemp().substring(0, 2), 16); if (beginIndex % 2 != 0) { // 奇数 +1 beginIndex++; } // 跳过 {长度}、{类型} = 4位 smsBean.setPduTelphone(smsBean.getTemp().substring(4, beginIndex + 4)); log.debug("PDU格式的发送方号码:" + smsBean.getPduTelphone()); String telphone = smsBean.getPduTelphone(); smsBean.setTemp(smsBean.getTemp().replaceAll(smsBean.getPduTelphone(), "").substring(8)); // 清除{PDU手机号}去除{长度}、{类型}、{类型}、{编码}的长度8位 StringBuilder builder = new StringBuilder(); for (int i = 0; i < telphone.length(); i += 2) { builder.append(telphone.charAt(i + 1)).append(telphone.charAt(i)); } return builder.toString().replaceAll("F", ""); }
3 . 提取短信发送时间
static String getDataTime(SmsBean smsBean) { log.debug("PDU格式发送时间:" + smsBean.getTemp().substring(0, 14)); //这里的02做法不太合理,由于现在是20XX年了,所以这里添加了这个,要是出现19XX的话就错误了。 String date = "02" + smsBean.getTemp().substring(0, 12); smsBean.setTemp(smsBean.getTemp().substring(16)); StringBuilder builder = new StringBuilder(); for (int i = 0; i < date.length(); i += 2) { builder.append(date.charAt(i + 1)).append(date.charAt(i)); } //这里调用了我写的另一个工具类时行日期格式转换。。 return DateUtils.format(builder.toString(), DatePattern.PATTERN_TWO.getPatten(), DatePattern.PATTERN_ONE.getPatten()); }
4. 提取短信内容
public static String getContent(SmsBean smsBean) {
log.debug("短信内容PDU格式:" + smsBean.getTemp()); // 得判断是否为多条短信发送过来,目前规范中没找到具体的这个说明,所以这里只能判断如果带这个 Matcher matcher2 = Pattern.compile("050003.{2}\\d{2}\\d{2}").matcher(smsBean.getTemp()); StringBuilder _beginStr = new StringBuilder(); while (matcher2.find()) { // 移除并分析追加短信头 String begin = matcher2.group(); if (begin != null) { begin = begin.replaceFirst("^050003.{2}", ""); if (begin.length() == 4) { _beginStr.append("(共").append(begin.substring(0, 2)).append("条,第"); _beginStr.append(begin.substring(2, 4)).append("条) "); } } break; } return _beginStr.append(UnicodeConversion.decodeUCS2(smsBean.getTemp().replaceFirst("050003.{2}\\d{2}\\d{2}", ""))).toString(); }
另外,发送时的格式与这又不一致了。。。