Jackson第二篇【从JSON字符串中取值】

第一篇咱们主要学习了实体与json的相互转换的问题,但是咱们需要的是数据 你转换18遍我取不到数据也是扯淡,那么今天咱们就一起学习一下如何从使用Jackson从Json字符串中取值。废话不说直接上代码( 注意第一篇里面的方法我都移到JsonProcessUtil里面了方便使用  )。

    从Json字符串中取出指定节点的值

 

Java代码  复制代码
  1. public static void getValue(TestVo vo) throws Exception {   
  2.     // 准备工作 传入vo请参照第一篇里面的实体。此处不再重新贴上代码 浪费大家时间   
  3.     ObjectMapper mapper = JsonProcessUtil.getMapperInstance(false);   
  4.     String voJson = JsonProcessUtil.toJson(vo);   
  5.     JsonNode node = mapper.readTree(voJson);// 这里的JsonNode和XML里面的Node很像   
  6.     System.out.println("readValueFromJson>>>" + node.get("voName").toString());// 获取voName   
  7.     // 输出结果:readValueFromJson>>>一个容器而已   
  8. }  
[java]  view plain copy
  1. public static void getValue(TestVo vo) throws Exception {  
  2.     // 准备工作 传入vo请参照第一篇里面的实体。此处不再重新贴上代码 浪费大家时间  
  3.     ObjectMapper mapper = JsonProcessUtil.getMapperInstance(false);  
  4.     String voJson = JsonProcessUtil.toJson(vo);  
  5.     JsonNode node = mapper.readTree(voJson);// 这里的JsonNode和XML里面的Node很像  
  6.     System.out.println("readValueFromJson>>>" + node.get("voName").toString());// 获取voName  
  7.     // 输出结果:readValueFromJson>>>一个容器而已  
  8. }  

    怎么样简单吧,但是这个作用不是很大啊!我想获取TestVo下面的Person的name的值 该怎么办呢?别着急,咱们走起!

 

Java代码  复制代码
  1. public static void getValue(TestVo vo) throws Exception {   
  2.     // 准备工作 传入vo请参照第一篇里面的实体。此处不再重新贴上代码 浪费大家时间   
  3.     ObjectMapper mapper = JsonProcessUtil.getMapperInstance(false);   
  4.     String voJson = JsonProcessUtil.toJson(vo);   
  5.     JsonNode node = mapper.readTree(voJson);// 这里的JsonNode和XML里面的Node很像   
  6.     node = node.get("pers");   
  7.     System.out.println("node是不是个集合:" + node.isArray());// 这个方法咱们后面会用到先让它给大家照个面   
  8.     for (int i = 0; i < node.size(); i++) {   
  9.         JsonNode childNode = node.get(i);   
  10.         System.out.println("readValueFromJson>>>" + childNode.get("name").toString());// 获取name   
  11.     }   
  12.     /*  
  13.      * 输出结果   
  14.      * node是不是个集合:true  
  15.      * readValueFromJson>>>张三   
  16.      * readValueFromJson>>>李四  
  17.      * readValueFromJson>>>王二麻子  
  18.      */  
  19. }  
[java]  view plain copy
  1. public static void getValue(TestVo vo) throws Exception {  
  2.     // 准备工作 传入vo请参照第一篇里面的实体。此处不再重新贴上代码 浪费大家时间  
  3.     ObjectMapper mapper = JsonProcessUtil.getMapperInstance(false);  
  4.     String voJson = JsonProcessUtil.toJson(vo);  
  5.     JsonNode node = mapper.readTree(voJson);// 这里的JsonNode和XML里面的Node很像  
  6.     node = node.get("pers");  
  7.     System.out.println("node是不是个集合:" + node.isArray());// 这个方法咱们后面会用到先让它给大家照个面  
  8.     for (int i = 0; i < node.size(); i++) {  
  9.         JsonNode childNode = node.get(i);  
  10.         System.out.println("readValueFromJson>>>" + childNode.get("name").toString());// 获取name  
  11.     }  
  12.     /* 
  13.      * 输出结果  
  14.      * node是不是个集合:true 
  15.      * readValueFromJson>>>张三  
  16.      * readValueFromJson>>>李四 
  17.      * readValueFromJson>>>王二麻子 
  18.      */  
  19. }  

 怎么样功能还行吧?如果这样你就满意了,那你也太容易满足了。现在我在项目中需要像xpath那样来查找某节点的值(不太了解xpath的童鞋别着急我以后会和大家一起学习下)。/a/b/c/d 这样找到某个节点的值。为了增加复杂度,我又给Person类里面增加 一个List<Person> childs;表示这个人的孩子那么咱们一往下需找的节点就多了……

  首先看下我准备的数据

 

Java代码  复制代码
  1. {"voName":"一个容器而已","pers":[{"name":"张三","age":46,"childs":[{"name":"小张三1","age":20,"childs":null},{"name":"小张三2","age":17,"childs":null}]},{"name":"李四","age":29,"childs":[{"name":"小李四1","age":20,"childs":null}]},{"name":"王二麻子","age":23,"childs":null}]}  
[java]  view plain copy
  1. {"voName":"一个容器而已","pers":[{"name":"张三","age":46,"childs":[{"name":"小张三1","age":20,"childs":null},{"name":"小张三2","age":17,"childs":null}]},{"name":"李四","age":29,"childs":[{"name":"小李四1","age":20,"childs":null}]},{"name":"王二麻子","age":23,"childs":null}]}  

  张三:有两个孩子  李四:有一个孩子  王二麻子:没有孩子

  我现在要做的就是把所有的孩子找出来。

  废话不多说直接看代码

 

Java代码  复制代码
  1. public static void main(String[] args) throws Exception {   
  2.     // 准备数据   
  3.     List<Person> pers = new ArrayList<Person>();   
  4.     List<Person> childs = new ArrayList<Person>();   
  5.     Person p = new Person("张三"46);   
  6.     childs.add(new Person("小张三1"20));   
  7.     childs.add(new Person("小张三2"17));   
  8.     p.setChilds(childs);   
  9.     pers.add(p);   
  10.     p = new Person("李四"29);   
  11.     childs = new ArrayList<Person>();   
  12.     childs.add(new Person("小李四1"20));   
  13.     p.setChilds(childs);   
  14.     pers.add(p);   
  15.     p = new Person("王二麻子"23);   
  16.     pers.add(p);   
  17.     TestVo vo = new TestVo("一个容器而已", pers);   
  18.     // 实体转JSON字符串   
  19.     String json = beanToJson(vo);   
  20.     Object[] obj = readValueFromJson(json, "pers:childs:name").toArray();   
  21.     System.out.println(Arrays.toString(obj));   
  22.     // 输出结果:[小张三1, 小张三2, 小李四1]   
  23. }  
[java]  view plain copy
  1. public static void main(String[] args) throws Exception {  
  2.     // 准备数据  
  3.     List<Person> pers = new ArrayList<Person>();  
  4.     List<Person> childs = new ArrayList<Person>();  
  5.     Person p = new Person("张三"46);  
  6.     childs.add(new Person("小张三1"20));  
  7.     childs.add(new Person("小张三2"17));  
  8.     p.setChilds(childs);  
  9.     pers.add(p);  
  10.     p = new Person("李四"29);  
  11.     childs = new ArrayList<Person>();  
  12.     childs.add(new Person("小李四1"20));  
  13.     p.setChilds(childs);  
  14.     pers.add(p);  
  15.     p = new Person("王二麻子"23);  
  16.     pers.add(p);  
  17.     TestVo vo = new TestVo("一个容器而已", pers);  
  18.     // 实体转JSON字符串  
  19.     String json = beanToJson(vo);  
  20.     Object[] obj = readValueFromJson(json, "pers:childs:name").toArray();  
  21.     System.out.println(Arrays.toString(obj));  
  22.     // 输出结果:[小张三1, 小张三2, 小李四1]  
  23. }  

    好玩吧,接下来咱们就一起看下我这个readValueFormJson是如何实现的呢?

 

Java代码  复制代码
  1. /**  
  2.  * 从json中读取tagPath处的值 tagPath用 :分隔  
  3.  *   
  4.  * @param json  
  5.  * @param tagPath  
  6.  * @return  
  7.  * @throws Exception  
  8.  */  
  9. public static List<String> readValueFromJson(String json, String tagPath) throws Exception {   
  10.     // 返回值   
  11.     List<String> value = new ArrayList<String>();   
  12.     if (CommonUtil.isEmpty(json) || (CommonUtil.isEmpty(tagPath))) {   
  13.         return value;   
  14.     }   
  15.     ObjectMapper mapper = CommonUtil.getMapperInstance(false);   
  16.     String[] path = tagPath.split(":");   
  17.     JsonNode node = mapper.readTree(json);   
  18.     getJsonValue(node, path, value, 1);   
  19.     return value;   
  20. }   
  21.   
  22. public static void getJsonValue(JsonNode node, String[] path, List<String> values, int nextIndex) {   
  23.     if (CommonUtil.isEmpty(node)) {   
  24.         return;   
  25.     }   
  26.     // 是路径的最后就直接取值   
  27.     if (nextIndex == path.length) {   
  28.         if (node.isArray()) {   
  29.             for (int i = 0; i < node.size(); i++) {   
  30.                 JsonNode child = node.get(i).get(path[nextIndex - 1]);   
  31.                 if (CommonUtil.isEmpty(child)) {   
  32.                     continue;   
  33.                 }   
  34.                 values.add(child.toString());   
  35.             }   
  36.         } else {   
  37.             JsonNode child = node.get(path[nextIndex - 1]);   
  38.             if (!CommonUtil.isEmpty(child)) {   
  39.                 values.add(child.toString);   
  40.             }   
  41.         }   
  42.         return;   
  43.     }   
  44.     // 判断是Node下是集合还是一个节点   
  45.     node = node.get(path[nextIndex - 1]);   
  46.     if (node.isArray()) {   
  47.         for (int i = 0; i < node.size(); i++) {   
  48.             getJsonValue(node.get(i), path, values, nextIndex + 1);   
  49.         }   
  50.     } else {   
  51.         getJsonValue(node, path, values, nextIndex + 1);   
  52.     }   
  53. }  
[java]  view plain copy
  1. /** 
  2.  * 从json中读取tagPath处的值 tagPath用 :分隔 
  3.  *  
  4.  * @param json 
  5.  * @param tagPath 
  6.  * @return 
  7.  * @throws Exception 
  8.  */  
  9. public static List<String> readValueFromJson(String json, String tagPath) throws Exception {  
  10.     // 返回值  
  11.     List<String> value = new ArrayList<String>();  
  12.     if (CommonUtil.isEmpty(json) || (CommonUtil.isEmpty(tagPath))) {  
  13.         return value;  
  14.     }  
  15.     ObjectMapper mapper = CommonUtil.getMapperInstance(false);  
  16.     String[] path = tagPath.split(":");  
  17.     JsonNode node = mapper.readTree(json);  
  18.     getJsonValue(node, path, value, 1);  
  19.     return value;  
  20. }  
  21.   
  22. public static void getJsonValue(JsonNode node, String[] path, List<String> values, int nextIndex) {  
  23.     if (CommonUtil.isEmpty(node)) {  
  24.         return;  
  25.     }  
  26.     // 是路径的最后就直接取值  
  27.     if (nextIndex == path.length) {  
  28.         if (node.isArray()) {  
  29.             for (int i = 0; i < node.size(); i++) {  
  30.                 JsonNode child = node.get(i).get(path[nextIndex - 1]);  
  31.                 if (CommonUtil.isEmpty(child)) {  
  32.                     continue;  
  33.                 }  
  34.                 values.add(child.toString());  
  35.             }  
  36.         } else {  
  37.             JsonNode child = node.get(path[nextIndex - 1]);  
  38.             if (!CommonUtil.isEmpty(child)) {  
  39.                 values.add(child.toString);  
  40.             }  
  41.         }  
  42.         return;  
  43.     }  
  44.     // 判断是Node下是集合还是一个节点  
  45.     node = node.get(path[nextIndex - 1]);  
  46.     if (node.isArray()) {  
  47.         for (int i = 0; i < node.size(); i++) {  
  48.             getJsonValue(node.get(i), path, values, nextIndex + 1);  
  49.         }  
  50.     } else {  
  51.         getJsonValue(node, path, values, nextIndex + 1);  
  52.     }  
  53. }  

     挺好吧,功能虽然实现了。但是童鞋们可能会对这个产生疑问,你他X的不是脱裤子放屁吗?先把Bean转换成Json再从Json中取值。效率低不说,而且还复杂。我直接通过get方法多好……如果你这样想了说明你是一个思考者!但是有种情况(本人在开发中遇到的)如果这个bean非常大 不同情况你需要的字段是不一样的 那你怎办? 根据情况判断if else if else if?如果有一百种情况呢?那不就累死了。所以在数据库里面配置不同业务情况从不同字段取值比较好,这样就会用到这种功能。另外附上:CommonUtil.isEmpty()方法

Java代码  复制代码
  1. /**  
  2.      * 判断对象是否为空  
  3.      *   
  4.      * @param obj  
  5.      * @return  
  6.      */  
  7.     public static boolean isEmpty(Object obj) {   
  8.         boolean result = true;   
  9.         if (obj == null) {   
  10.             return true;   
  11.         }   
  12.         if (obj instanceof String) {   
  13.             result = (obj.toString().trim().length() == 0) || obj.toString().trim().equals("null");   
  14.         } else if (obj instanceof Collection) {   
  15.             result = ((Collection) obj).size() == 0;   
  16.         } else {   
  17.             result = ((obj == null) || (obj.toString().trim().length() < 1)) ? true : false;   
  18.         }   
  19.         return result;   
  20.     }  
[java]  view plain copy
  1. /** 
  2.      * 判断对象是否为空 
  3.      *  
  4.      * @param obj 
  5.      * @return 
  6.      */  
  7.     public static boolean isEmpty(Object obj) {  
  8.         boolean result = true;  
  9.         if (obj == null) {  
  10.             return true;  
  11.         }  
  12.         if (obj instanceof String) {  
  13.             result = (obj.toString().trim().length() == 0) || obj.toString().trim().equals("null");  
  14.         } else if (obj instanceof Collection) {  
  15.             result = ((Collection) obj).size() == 0;  
  16.         } else {  
  17.             result = ((obj == null) || (obj.toString().trim().length() < 1)) ? true : false;  
  18.         }  
  19.         return result;  
  20.     }  

 注意:JsonNode 提供了很多种取值的方法,但是为了通用我选择了toString()因为getTextValue(),getIntValue()……一些方法只能返回指定类型的值,这样不能达到通用。而且有一种情况如果我想要的是整个节点下面所有子节点的值呢?如果使用上面的方法就没办法了,只能使用toString()这时候它返回的是整个json字符串。注意:toString()方法如果只返回某一个属性的值 如:name  而不是一整个大节点如:name,age,gender,那么它会在返回的字符串上加双引号,用的时候注意去掉开始结束的双引号。

今天就学习到这,下篇文章我将与大家一起探讨一下为什么 我创建一个ObjectMapper对象要搞的那么复杂!不直接new。

你可能感兴趣的:(Jackson第二篇【从JSON字符串中取值】)