国内一个据说专业外包户的xxxInfo公司,再做所谓的集中话,听说项目中标N个亿,在这就不细说了。项目具体实施情况就不说了,说说最近在遇到的一个坑人的事情以及解决办法哈。所在项目组,不对应该说是小作坊,根据甲方爸爸的要求需要接入该集中的项目,其实就是一个统一的API网关(当然现在本来就是用的统一API网关,只是这次的更集中话了而已)。因此需要我们配合测试API是否正常,数据是否正确。本来是个很简单的事情但是在测试的时候发现返回结果的JSON格式有差异,这就比较坑了,测试要求是需要给对方接口人提供差异点,比如说原本JSON的中X_RESULT_CODE但是现在是X_CODE,注入此类的问题很多。由于线上接口比较多而已返回JSON格式比较复杂,单靠眼里对比那是相当坑人的。即便先格式化再排序对比还是需要靠人眼去识别。
基于此说说JSON格式对于吧,其实但对比格式还是比较容易实现的,就是将所有的键值全体提取处理,然后做差异对比即可。当然再使用的操作需要处理数组类型的值。可以将所有的键值全体提取成js访问格式分为如下俩种
data.iddata.id.valuedata.list[x]data.list[x].val//以上俩种的N多中变种格式
另外还有一种比较常见后台API接口中的变种方式JSON的value值是个字符串而非json对象,但是这个字符串能通过解码或者不解码可以直接转换成一个JSON对象。这种情况也需要单独处理。Talk is cheap. Show me the code!!!
/** * 利用fastjson 实现的简单json键值提取 * @param bkey * @param node * @param dealStringJson * @return */ public static Set<String> jsonObjectKeySet(String bkey, JSONObject node, boolean dealStringJson) { Set<String> keySet = new HashSet<String>(); Set<String> strings = node.keySet(); for (String key : strings) { Object value = node.get(key); String myKey = key; if (!StringUtils.isBlank(bkey)) { myKey = bkey + "." + key; } keySet.add(myKey); if (value instanceof JSONObject || value instanceof JSONArray) { keySet.addAll(jsonKeySet(myKey, value, dealStringJson)); } if (dealStringJson && value instanceof String) { String strJson = (String) value; if (strJson.startsWith("{") && strJson.endsWith("}")) { keySet.addAll(jsonKeySet(myKey, JSONObject.parseObject(strJson), dealStringJson)); } if (strJson.startsWith("[") && strJson.endsWith("]")) { System.out.println("@@@" + strJson); JSONArray parseObject = JSONArray.parseArray(strJson); keySet.addAll(jsonKeySet(myKey, parseObject, dealStringJson)); } } } return keySet; } public static Set<String> jsonKeySet(String bkey, Object node, boolean dealStringJson) { Set<String> keySet = new HashSet<String>(); if (node instanceof JSONObject) { keySet.addAll(jsonObjectKeySet(bkey, (JSONObject) node, dealStringJson)); } if (node instanceof JSONArray) { Iterator<Object> iterator = ((JSONArray) node).iterator(); while (iterator.hasNext()) { Object next = iterator.next(); String bsKey = bkey + "[x]"; if (next instanceof JSONObject) { keySet.addAll(jsonObjectKeySet(bsKey, (JSONObject) next, dealStringJson)); } } } return keySet; } /** * 利用Jackson实现简单的json键值提取 * @param bKey * @param node * @return * @throws IOException */ public static Set<String> jsonLeaf(String bKey, JsonNode node) throws IOException { Set<String> keySet = new HashSet<String>(); if (node.isValueNode()) { String strJson = node.toString(); if (strJson.startsWith("{") && strJson.endsWith("}")) { ObjectMapper jackson = new ObjectMapper(); JsonNode nodeStr = jackson.readTree(strJson); keySet.addAll(jsonLeaf(bKey, nodeStr)); } if (strJson.startsWith("[") && strJson.endsWith("]")) { ObjectMapper jackson = new ObjectMapper(); JsonNode nodeStr = jackson.readTree(strJson); keySet.addAll(jsonLeaf(bKey + "[x]", nodeStr)); } return keySet; } if (node.isObject()) { Iterator<Map.Entry<String, JsonNode>> it = node.fields(); while (it.hasNext()) { Map.Entry<String, JsonNode> entry = it.next(); String key = entry.getKey(); if (!StringUtils.isBlank(bKey)) { key = bKey + "." + key; } keySet.add(key); JsonNode value = entry.getValue(); if (value.isContainerNode()) { keySet.addAll(jsonLeaf(key, value)); } } } if (node.isArray()) { Iterator<JsonNode> it = node.iterator(); while (it.hasNext()) { keySet.addAll(jsonLeaf(bKey + "[x]", it.next())); } } return keySet; }
调用实例
JSONObject jsonObject = JSONObject.parseObject(json);Set<String> strings = jsonKeySet("", jsonObject, true);Set<String> sortSet = new TreeSet<String>(new Comparator<String>() { @Override public int compare(String o1, String o2) { return o1.compareTo(o2); }});sortSet.addAll(strings);for (String key : sortSet) { System.out.println(key);}
剩下的就是利用set很简单的就能获取到差异了。然后配置模板自动回复微信了,然后剩下的就是可以摸鱼了。
当然还有一个问题也可以利用该方法进一步完成,获取到JSON键值的差异以后可以反向生成实体对象或者生成字段映射关系,只是这样效率可能有些低,臃肿就是这样产生的。当然为了追求完美可以level并新开一个v,通过level实现新老转换,新v全部统一,外围逐步同步到新v上即可。