前言
电信是最早开通NB业务的运营商,两年前我们公司就已经着手NB表具的开发工作,当时联通和移动的NB业务还没有开展,自然没有考虑兼容的问题,现在下定决心重新做一套肯定是要做一套兼容的程序!那么,收发机制完全不同的移动NB和电信NB怎样才能实现兼容呢?
找到不同点:
移动是直接向平台发送16进制字符串,平台将其自动转为16进制码流向表具发送16进制码流,如下图所示
电信是通过profile文件向平台发送配置文件中的json,再由编解码插件转为16进制码流,再发向表具,如下图所示
那我们只需要将他们不同的地方做成一样的就可以完成兼容了!即应用应该向平台发送16进制码流就可以了!!应用发送16进制码流,就不需要编解码插件了,但是它又必须存在,所以我们需要插件做“没有意义”的转换!!
实现准备
profile中的service有两个列表,一个是属性列表:AcceptService,一个是命令列表:CommandService
属性列表用来接收编解码插件中decode方法解码出来的属性
命令列表用来传入编解码插件中encode方法需要传入的属性
我们在定义profile的时候要把他们尽量分开
发送命令
CommandService 中只编辑命令列表,因为我们要做到兼容,就是不管下发什么命令,我都只发送一个16进制码流,只是根据不同的命令下发的码流不同而已,所以我们只定义一个send命令,命令中只需定义一个value属性用来存放我们要发送的码流!
与之对应的是编解码插件中的encode方法,而encode方法传入一个ObjectNode(json)返回一个byte数组,我们在encode中要做的就是将json中的value字符串拿出来转换成byte数组
public byte[] encode(ObjectNode input) throws Exception { String hexString = input.get("paras").get("value").asText(); List<String> list = new ArrayList<String>(); for (int i = 0; i < hexString.length() - 1; i += 2) { list.add(hexString.substring(i, i + 2)); } byte[] byteArray = new byte[list.size()]; for (int i = 0; i < list.size(); i++) { int dec_num = (int) Long.parseLong(list.get(i), 16); byteArray[i] = (byte) dec_num; } return byteArray; }
接收上传数据
AcceptService 中只编辑属性列表,同样我们的编解码插件只是起一个类型转换的作用,只需在该服务下定义一个value属性,用来接收从编解码插件中返回的字符串
与之对应的是编解码插件中的decode方法,decode方法传入一个byte数组返回一个ObjectNode(json),我们在decode中要做的就是将byte数组转换成字符串放入AcceptService的value属性中
public ObjectNode decode(byte[] binaryData) throws Exception { StringBuffer result = new StringBuffer(); System.err.println( binaryData.length); for (int i = 0; i < binaryData.length; i++) { String hex = String.format("%02x", Byte.toUnsignedInt(binaryData[i])); result.append(hex); } //组装body体,只能为ObjectNode对象 ObjectMapper mapper = new ObjectMapper(); ObjectNode root = mapper.createObjectNode(); ArrayNode arrynode = mapper.createArrayNode(); ObjectNode batteryNode = mapper.createObjectNode(); batteryNode.put("serviceId", "AcceptService"); ObjectNode batteryData = mapper.createObjectNode(); batteryData.put("value", result.toString()); batteryNode.put("serviceData", batteryData); arrynode.add(batteryNode); root.put("msgType", "deviceReq"); root.put("data", arrynode); return root; }
测试
如下图所示新增一个虚拟设备
选中CommandService服务下的send命令,在value中填入16进制字符串:
如上图所示,我们完成了发送什么,设备端就接收到什么,测试接收服务在上图“请输入十六进制码流”的窗体没输入设备上传的数据,在应用模拟器中就能够看到接受到的内容!!
总结
如上我们就实现了电信NB和移动NB的兼容,即他们都是应用根据不同的业务场景,编辑不同的16进制字符串,然后向平台发送一个16进制字符串,然后平台通过转换向产品发送一个16进制码流;接收数据都是接收16进制字符串然后解析字符串得到想要的数据!
两种机制各有好处,作为一个中立的开发者来说:论开发效率,移动的模式甩电信10条街 - - ||