对于json格式的数据解析现在越来越多了,之前介绍了两种:fastjson和net.sf.json解析。

今天又有一个jackson解析。不过相对于之前两种,这种感觉稍微笨拙些。呵呵,还是了解下吧;

转:http://blog.csdn.NET/legend_x/article/details/31767083

jackson的介绍

http://www.cnblogs.com/winner-0715/p/6109225.html

Java下常见的Json类库有Gson、JSON-lib和Jackson等,Jackson相对来说比较高效,在项目中主要使用Jackson进行JSON和Java对象转换,下面给出一些Jackson的JSON操作方法。

一、准备工作

Jackson有1.x系列和2.x系列,2.x系列有3个jar包需要下载:
jackson-core-2.2.3.jar(核心jar包)
jackson-annotations-2.2.3.jar(该包提供Json注解支持)
jackson-databind-2.2.3.jar

  1. /**
  2. * ObjectMapper是JSON操作的核心,Jackson的所有JSON操作都是在ObjectMapper中实现。
  3. * ObjectMapper有多个JSON序列化的方法,可以把JSON字符串保存File、OutputStream等不同的介质中。
  4. * writeValue(File arg0, Object arg1)把arg1转成json序列,并保存到arg0文件中。
  5. * writeValue(OutputStream arg0, Object arg1)把arg1转成json序列,并保存到arg0输出流中。
  6. * writeValueAsBytes(Object arg0)把arg0转成json序列,并把结果输出成字节数组。
  7. * writeValueAsString(Object arg0)把arg0转成json序列,并把结果输出成字符串。
  8. */

Jackson 框架,轻易转换JSON

Jackson可以轻松的将Java对象转换成json对象和xml文档,同样也可以将json、xml转换成Java对象。

前面有介绍过json-lib这个框架,在线博文:http://www.cnblogs.com/hoojo/archive/2011/04/21/2023805.html

相比json-lib框架,Jackson所依赖的jar包较少,简单易用并且性能也要相对高些。而且Jackson社区相对比较活跃,更新速度也比较快。

一、准备工作

1、 下载依赖库jar包

Jackson的jar all下载地址:http://jackson.codehaus.org/1.7.6/jackson-all-1.7.6.jar

然后在工程中导入这个jar包即可开始工作

官方示例:http://wiki.fasterxml.com/JacksonInFiveMinutes

因为下面的程序是用junit测试用例运行的,所以还得添加junit的jar包。版本是junit-4.2.8

如果你需要转换xml,那么还需要stax2-api.jar

2、 测试类基本代码如下

  1. package com.hoo.test;
  1.  
  1. import java.io.IOException;
  1. import java.io.StringWriter;
  1. import java.util.ArrayList;
  1. import java.util.HashMap;
  1. import java.util.Iterator;
  1. import java.util.LinkedHashMap;
  1. import java.util.List;
  1. import java.util.Map;
  1. import java.util.Set;
  1. import org.codehaus.jackson.JsonEncoding;
  1. import org.codehaus.jackson.JsonGenerationException;
  1. import org.codehaus.jackson.JsonGenerator;
  1. import org.codehaus.jackson.JsonParseException;
  1. import org.codehaus.jackson.map.JsonMappingException;
  1. import org.codehaus.jackson.map.ObjectMapper;
  1. import org.codehaus.jackson.node.JsonNodeFactory;
  1. import org.codehaus.jackson.xml.XmlMapper;
  1. import org.junit.After;
  1. import org.junit.Before;
  1. import org.junit.Test;
  1. import com.hoo.entity.AccountBean;
  1.  
  1. /**
  1. * <b>function:</b>Jackson java对象转换成JSON字符串,也可以将JSON字符串转换成java对象
  1. * jar-lib-version: jackson-all-1.6.2
  1. * jettison-1.0.1
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午04:54:53
  1. * @file JacksonTest.java
  1. * @package com.hoo.test
  1. * @project Spring3
  1. * @blog http://blog.csdn.net/IBM_hoojo
  1. * @email hoojo_@126.com
  1. * @version 1.0
  1. */
  1. @SuppressWarnings("unchecked")
  1. public class JacksonTest {
  1. private JsonGenerator jsonGenerator = null;
  1. private ObjectMapper objectMapper = null;
  1. private AccountBean bean = null;
  1. @Before
  1. public void init() {
  1. bean = new AccountBean();
  1. bean.setAddress("china-Guangzhou");
  1. bean.setEmail("hoojo_@126.com");
  1. bean.setId(1);
  1. bean.setName("hoojo");
  1. objectMapper = new ObjectMapper();
  1. try {
  1. jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }
  1. @After
  1. public void destory() {
  1. try {
  1. if (jsonGenerator != null) {
  1. jsonGenerator.flush();
  1. }
  1. if (!jsonGenerator.isClosed()) {
  1. jsonGenerator.close();
  1. }
  1. jsonGenerator = null;
  1. objectMapper = null;
  1. bean = null;
  1. System.gc();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }
  1. }

3、 所需要的JavaEntity

  1. package com.hoo.entity;
  1.  
  1. public class AccountBean {
  1. private int id;
  1. private String name;
  1. private String email;
  1. private String address;
  1. private Birthday birthday;
  1. //getter、setter
  1. @Override
  1. public String toString() {
  1. return this.name + "#" + this.id + "#" + this.address + "#" + this.birthday + "#" + this.email;
  1. }
  1. }

Birthday

  1. package com.hoo.entity;
  1.  
  1. public class Birthday {
  1. private String birthday;
  1. public Birthday(String birthday) {
  1. super();
  1. this.birthday = birthday;
  1. }
  1.  
  1. //getter、setter
  1.  
  1. public Birthday() {}
  1. @Override
  1. public String toString() {
  1. return this.birthday;
  1. }
  1. }

二、Java对象转换成JSON

1、 JavaBean(Entity/Model)转换成JSON

  1. /**
  1. * <b>function:</b>将java对象转换成json字符串
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:01:10
  1. */
  1. @Test
  1. public void writeEntityJSON() {
  1. try {
  1. System.out.println("jsonGenerator");
  1. //writeObject可以转换java对象,eg:JavaBean/Map/List/Array等
  1. jsonGenerator.writeObject(bean);
  1. System.out.println();
  1. System.out.println("ObjectMapper");
  1. //writeValue具有和writeObject相同的功能
  1. objectMapper.writeValue(System.out, bean);
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

运行后结果如下:

  1. jsonGenerator
  1. {"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"}
  1. ObjectMapper
  1. {"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"}

上面分别利用JsonGenerator的writeObject方法和ObjectMapper的writeValue方法完成对Java对象的转换,二者传递的参数及构造的方式不同;JsonGenerator的创建依赖于ObjectMapper对象。也就是说如果你要使用JsonGenerator来转换JSON,那么你必须创建一个ObjectMapper。但是你用ObjectMapper来转换JSON,则不需要JSONGenerator。

objectMapper的writeValue方法可以将一个Java对象转换成JSON。这个方法的参数一,需要提供一个输出流,转换后可以通过这个流来输出转换后的内容。或是提供一个File,将转换后的内容写入到File中。当然,这个参数也可以接收一个JSONGenerator,然后通过JSONGenerator来输出转换后的信息。第二个参数是将要被转换的Java对象。如果用三个参数的方法,那么是一个Config。这个config可以提供一些转换时的规则,过指定的Java对象的某些属性进行过滤或转换等。

2、 将Map集合转换成Json字符串

  1. /**
  1. * <b>function:</b>将map转换成json字符串
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:05:26
  1. */
  1. @Test
  1. public void writeMapJSON() {
  1. try {
  1. Map<String, Object> map = new HashMap<String, Object>();
  1. map.put("name", bean.getName());
  1. map.put("account", bean);
  1. bean = new AccountBean();
  1. bean.setAddress("china-Beijin");
  1. bean.setEmail("hoojo@qq.com");
  1. map.put("account2", bean);
  1. System.out.println("jsonGenerator");
  1. jsonGenerator.writeObject(map);
  1. System.out.println("");
  1. System.out.println("objectMapper");
  1. objectMapper.writeValue(System.out, map);
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

转换后结果如下:

  1. jsonGenerator
  1. {"account2":{"address":"china-Beijin","name":null,"id":0,"birthday":null,"email":"hoojo@qq.com"},"name":"hoojo",
  1. "account":{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"}}
  1. objectMapper
  1. {"account2":{"address":"china-Beijin","name":null,"id":0,"birthday":null,"email":"hoojo@qq.com"},"name":"hoojo",
  1. "account":{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"}}

3、 将List集合转换成json

  1. /**
  1. * <b>function:</b>将list集合转换成json字符串
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:05:59
  1. */
  1. @Test
  1. public void writeListJSON() {
  1. try {
  1. List<AccountBean> list = new ArrayList<AccountBean>();
  1. list.add(bean);
  1. bean = new AccountBean();
  1. bean.setId(2);
  1. bean.setAddress("address2");
  1. bean.setEmail("email2");
  1. bean.setName("haha2");
  1. list.add(bean);
  1. System.out.println("jsonGenerator");
  1. //list转换成JSON字符串
  1. jsonGenerator.writeObject(list);
  1. System.out.println();
  1. System.out.println("ObjectMapper");
  1. //用objectMapper直接返回list转换成的JSON字符串
  1. System.out.println("1###" + objectMapper.writeValueAsString(list));
  1. System.out.print("2###");
  1. //objectMapper list转换成JSON字符串
  1. objectMapper.writeValue(System.out, list);
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

结果如下:

  1. jsonGenerator
  1. [{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"},
  1. {"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]
  1. ObjectMapper
  1. 1###[{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"},
  1. {"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]
  1. 2###[{"address":"china-Guangzhou","name":"hoojo","id":1,"birthday":null,"email":"hoojo_@126.com"},
  1. {"address":"address2","name":"haha2","id":2,"birthday":null,"email":"email2"}]

外面就是多了个[]中括号;同样Array也可以转换,转换的JSON和上面的结果是一样的,这里就不再转换了。~.~

4、下面来看看jackson提供的一些类型,用这些类型完成json转换;如果你使用这些类型转换JSON的话,那么你即使没有JavaBean(Entity)也可以完成复杂的Java类型的JSON转换。下面用到这些类型构建一个复杂的Java对象,并完成JSON转换。

  1. @Test
  1. public void writeOthersJSON() {
  1. try {
  1. String[] arr = { "a", "b", "c" };
  1. System.out.println("jsonGenerator");
  1. String str = "hello world jackson!";
  1. //byte
  1. jsonGenerator.writeBinary(str.getBytes());
  1. //boolean
  1. jsonGenerator.writeBoolean(true);
  1. //null
  1. jsonGenerator.writeNull();
  1. //float
  1. jsonGenerator.writeNumber(2.2f);
  1. //char
  1. jsonGenerator.writeRaw("c");
  1. //String
  1. jsonGenerator.writeRaw(str, 5, 10);
  1. //String
  1. jsonGenerator.writeRawValue(str, 5, 5);
  1. //String
  1. jsonGenerator.writeString(str);
  1. jsonGenerator.writeTree(JsonNodeFactory.instance.POJONode(str));
  1. System.out.println();
  1. //Object
  1. jsonGenerator.writeStartObject();//{
  1. jsonGenerator.writeObjectFieldStart("user");//user:{
  1. jsonGenerator.writeStringField("name", "jackson");//name:jackson
  1. jsonGenerator.writeBooleanField("sex", true);//sex:true
  1. jsonGenerator.writeNumberField("age", 22);//age:22
  1. jsonGenerator.writeEndObject();//}
  1. jsonGenerator.writeArrayFieldStart("infos");//infos:[
  1. jsonGenerator.writeNumber(22);//22
  1. jsonGenerator.writeString("this is array");//this is array
  1. jsonGenerator.writeEndArray();//]
  1. jsonGenerator.writeEndObject();//}
  1. AccountBean bean = new AccountBean();
  1. bean.setAddress("address");
  1. bean.setEmail("email");
  1. bean.setId(1);
  1. bean.setName("haha");
  1. //complex Object
  1. jsonGenerator.writeStartObject();//{
  1. jsonGenerator.writeObjectField("user", bean);//user:{bean}
  1. jsonGenerator.writeObjectField("infos", arr);//infos:[array]
  1. jsonGenerator.writeEndObject();//}
  1. } catch (Exception e) {
  1. e.printStackTrace();
  1. }
  1. }

运行后,结果如下:

  1. jsonGenerator
  1. "aGVsbG8gd29ybGQgamFja3NvbiE=" true null 2.2c world jac worl "hello world jackson!" "hello world jackson!"
  1. {"user":{"name":"jackson","sex":true,"age":22},"infos":[22,"this is array"]}
  1. {"user":{"address":"address","name":"haha","id":1,"birthday":null,"email":"email"},"infos":["a","b","c"]}

怎么样?构造的json字符串和输出的结果是一致的吧。关键看懂用JSONGenerator提供的方法,完成一个Object的构建。

三、JSON转换成Java对象

1、 将json字符串转换成JavaBean对象

  1. @Test
  1. public void readJson2Entity() {
  1. String json = "{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}";
  1. try {
  1. AccountBean acc = objectMapper.readValue(json, AccountBean.class);
  1. System.out.println(acc.getName());
  1. System.out.println(acc);
  1. } catch (JsonParseException e) {
  1. e.printStackTrace();
  1. } catch (JsonMappingException e) {
  1. e.printStackTrace();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

很简单,用到了ObjectMapper这个对象的readValue这个方法,这个方法需要提供2个参数。第一个参数就是解析的JSON字符串,第二个参数是即将将这个JSON解析吃什么Java对象,Java对象的类型。当然,还有其他相同签名方法,如果你有兴趣可以一一尝试使用方法,当然使用的方法和当前使用的方法大同小异。运行后,结果如下:

  1. haha
  1. haha#1#address#null#email

2、 将json字符串转换成List<Map>集合

  1. /**
  1. * <b>function:</b>json字符串转换成list<map>
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:12:01
  1. */
  1. @Test
  1. public void readJson2List() {
  1. String json = "[{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+
  1. "{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}]";
  1. try {
  1. List<LinkedHashMap<String, Object>> list = objectMapper.readValue(json, List.class);
  1. System.out.println(list.size());
  1. for (int i = 0; i < list.size(); i++) {
  1. Map<String, Object> map = list.get(i);
  1. Set<String> set = map.keySet();
  1. for (Iterator<String> it = set.iterator();it.hasNext();) {
  1. String key = it.next();
  1. System.out.println(key + ":" + map.get(key));
  1. }
  1. }
  1. } catch (JsonParseException e) {
  1. e.printStackTrace();
  1. } catch (JsonMappingException e) {
  1. e.printStackTrace();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

尝试过将上面的JSON转换成List,然后List中存放AccountBean,但结果失败了。但是支持Map集合。因为你转成List.class,但是不知道List存放何种类型。只好默然Map类型。因为所有的对象都可以转换成Map结合,运行后结果如下:

  1. 2
  1. address:address2
  1. name:haha2
  1. id:2
  1. email:email2
  1. address:address
  1. name:haha
  1. id:1
  1. email:email

3、 Json字符串转换成Array数组,由于上面的泛型转换不能识别到集合中的对象类型。所有这里用对象数组,可以解决这个问题。只不过它不再是集合,而是一个数组。当然这个不重要,你可以用Arrays.asList将其转换成List即可。

  1. /**
  1. * <b>function:</b>json字符串转换成Array
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:14:01
  1. */
  1. @Test
  1. public void readJson2Array() {
  1. String json = "[{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+
  1. "{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}]";
  1. try {
  1. AccountBean[] arr = objectMapper.readValue(json, AccountBean[].class);
  1. System.out.println(arr.length);
  1. for (int i = 0; i < arr.length; i++) {
  1. System.out.println(arr[i]);
  1. }
  1. } catch (JsonParseException e) {
  1. e.printStackTrace();
  1. } catch (JsonMappingException e) {
  1. e.printStackTrace();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

运行后的结果:

  1. 2
  1. haha2#2#address2#null#email2
  1. haha#1#address#null#email

4、 Json字符串转换成Map集合

  1. /**
  1. * <b>function:</b>json字符串转换Map集合
  1. * @author hoojo
  1. * @createDate Nov 27, 2010 3:00:06 PM
  1. */
  1. @Test
  1. public void readJson2Map() {
  1. String json = "{\"success\":true,\"A\":{\"address\": \"address2\",\"name\":\"haha2\",\"id\":2,\"email\":\"email2\"},"+
  1. "\"B\":{\"address\":\"address\",\"name\":\"haha\",\"id\":1,\"email\":\"email\"}}";
  1. try {
  1. Map<String, Map<String, Object>> maps = objectMapper.readValue(json, Map.class);
  1. System.out.println(maps.size());
  1. Set<String> key = maps.keySet();
  1. Iterator<String> iter = key.iterator();
  1. while (iter.hasNext()) {
  1. String field = iter.next();
  1. System.out.println(field + ":" + maps.get(field));
  1. }
  1. } catch (JsonParseException e) {
  1. e.printStackTrace();
  1. } catch (JsonMappingException e) {
  1. e.printStackTrace();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

运行后结果如下:

  1. 3
  1. success:true
  1. A:{address=address2, name=haha2, id=2, email=email2}
  1. B:{address=address, name=haha, id=1, email=email}

四、JacksonXML的支持

Jackson也可以完成java对象到xml的转换,转换后的结果要比json-lib更直观,不过它依赖于stax2-api.jar这个jar包。

  1. /**
  1. * <b>function:</b>java对象转换成xml文档
  1. * 需要额外的jar stax2-api.jar
  1. * @author hoojo
  1. * @createDate 2010-11-23 下午06:11:21
  1. */
  1. @Test
  1. public void writeObject2Xml() {
  1. //stax2-api-3.0.2.jar
  1. System.out.println("XmlMapper");
  1. XmlMapper xml = new XmlMapper();
  1. try {
  1. //javaBean转换成xml
  1. //xml.writeValue(System.out, bean);
  1. StringWriter sw = new StringWriter();
  1. xml.writeValue(sw, bean);
  1. System.out.println(sw.toString());
  1. //List转换成xml
  1. List<AccountBean> list = new ArrayList<AccountBean>();
  1. list.add(bean);
  1. list.add(bean);
  1. System.out.println(xml.writeValueAsString(list));
  1. //Map转换xml文档
  1. Map<String, AccountBean> map = new HashMap<String, AccountBean>();
  1. map.put("A", bean);
  1. map.put("B", bean);
  1. System.out.println(xml.writeValueAsString(map));
  1. } catch (JsonGenerationException e) {
  1. e.printStackTrace();
  1. } catch (JsonMappingException e) {
  1. e.printStackTrace();
  1. } catch (IOException e) {
  1. e.printStackTrace();
  1. }
  1. }

运行上面的方法,结果如下:

  1. XmlMapper
  1. <unknown><address>china-Guangzhou</address><name>hoojo</name><id>1</id><birthday/><email>hoojo_@126.com</email></unknown>
  1. <unknown><unknown><address>china-Guangzhou</address><name>hoojo</name><id>1</id><birthday/><email>hoojo_@126.com</email></unknown>
  1. <email><address>china-Guangzhou</address><name>hoojo</name><id>1</id><birthday/><email>hoojo_@126.com</email></email></unknown>
  1. <unknown><A><address>china-Guangzhou</address><name>hoojo</name><id>1</id><birthday/><email>hoojo_@126.com</email></A>
  1. <B><address>china-Guangzhou</address><name>hoojo</name><id>1</id><birthday/><email>hoojo_@126.com</email></B></unknown>

看结果,根节点都是unknown 这个问题还没有解决,由于根节点没有转换出来,所有导致解析xml到Java对象,也无法完成。

1、空值转换-异常情况:

  1. Can not instantiate value of type [map type; class java.util.HashMap, [simple type, class java.lang.String] -> [simple type, class java.lang.String]] from JSON String; no single-String constructor/factory method

解决方法,在ObjectMapper配置中增加:

  1. mapper.configure(DeserializationConfig.Feature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT, true) ;

2、转义字符-异常情况:

  1. org.codehaus.jackson.JsonParseException: Illegal unquoted character ((CTRL-CHAR, code 9)): has to be escaped using backslash to be included in string value
  2. at [Source: java.io.StringReader@10cfc2e3; line: 1, column: 2461]

解决办法,在ObjectMapper配置中增加:

  1. mapper.configure(Feature.ALLOW_UNQUOTED_CONTROL_CHARS, true) ;
 

jackson在http接口测试环境中的应用

蒋壮
  发布:2013-04-05 22:04   分类:接口测试文章分类框架分享   阅读:1,614 views   3条评论  

这里先简单的做下jackson的介绍:jackson可以很方便并且高效的将json对象转化成Java对象,也可以将Java对象转换成json对象和xml文档。Jackson有几个显著的特点:性能高,对于复杂对象和大数据量对象的序列化和反序列化,表现出比同类工具包(如gson,json-lib等)更优的性能;功能强,绑定所有JDK中的通用类以及Java Bean Class,Collection,Map和Enum;无依赖,除了JDK外无任何他依赖;完全开源,并且有丰富的文档和用户群体,便于自学。

在进行http请求的接口测试的过程中,我们需要灵活方便的处理http请求返回数据,使json串反序列化成为java对象,这样我们才能方便地对返回数据进行校验。如何试用jackson进行反序列化,网上内容很多,所以在此补赘述。我想重点说下在接口测试过程中需要注意的点和可能会遇到的问题的解决办法。

【注意点1】: 尽可能使用开发人员使用的数据对象类作为反序列化时的mapper类。

通常情况下,浏览器在发送http请求向服务端应用请求数据的时候,客户端应用都会依赖一个服务端的client包,以便于反序列化json数据后,形成用于vm模板的渲染所需要的java对象。这样就为我们测试http接口提供了便利,我们只需要在我们测试应用中依赖这个client的jar包,就可以方便的运用这些对象去组装我们的mapper。

【注意点2】: 当开发人员提供的数据对象类“不好用”的时候,需要自行创建pojo类,或者使用JDK的原生数据对象。

ObjectMapper初始化配置的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
ObjectMapper objectMapper = newObjectMapper();
 
//配置为true表示mapper接受只有一个元素的数组的反序列化
 
objectMapper.configure(Feature.ACCEPT_SINGLE_VALUE_AS_ARRAY,true);
 
//配置为false表示mapper在遇到mapper对象中存在json对象中没有的数据变量时不报错,可以进行反序列化
 
objectMapper.configure(Feature.FAIL_ON_UNKNOWN_PROPERTIES,false);
 
//新版的jackson设置mapper的方法,功能同上
 
objectMapper.getDeserializationConfig().without(DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES);
 
//定义针对日期类型的反序列化时的数据格式
 
objectMapper.getDeserializationConfig().setDateFormat(newSimpleDateFormat("yyyy-MM-dd HH:mm:ss"));

下面是处理json的工具类代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
publicstatic <T> T getObjectFromJsonString(String jsonString, TypeReference<T> tr) {
 
        if(jsonString == null|| "".equals(jsonString)) {
 
            returnnull;
 
        }else{
 
            try{
 
                return(T) objectMapper.readValue(jsonString, tr);
 
            }catch(Exception e) {
 
                e.printStackTrace();
 
            }
 
        }
 
        returnnull;
 
}

此方法采用了泛型,是为了抽象json返回对象的数据类型,以便后续开发通用的测试验证类

下面是调用类的方法的代码:

1
actDO = JsonUtils.getObjectFromJsonString(jsonStr, newTypeReference<BaseResponseModel<PublishCommentResultMapper>>() {});

或者

1
actDO = JsonUtils.getObjectFromJsonString(jsonStr, newTypeReference<BaseResponseModel<Map<String, String>>>() {});

下面是BaseResponseModel的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
publicclass BaseResponseModel<T> implementsSerializable{
 
    privatestatic final long serialVersionUID = 5393610697317077173L;
 
    privateList<T> result;
 
    privateboolean isSuccess;
 
    privateString errorMsg;
 
    privateint errorCode;
 
    privateInteger total;
 
    publicboolean isSuccess() {
 
       returnisSuccess;
 
    }
 
    publicvoid setIsSuccess(booleanisSuccess) {
 
       this.isSuccess = isSuccess;
 
    }
 
    //其余getter/setter省略
}
  1. BaseResponseModel封装了http请求返回的通用成员变量,例如:ErrorCode,IsSuccess,Total

Mapper对象实例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
publicclass InfoAndRateMapper implementsSerializable{
 
    privatestatic final long serialVersionUID = 1L;
 
    privateList<BrandContentInfoDO> info;
 
    privateList<Rate> rate;
 
    publicList<BrandContentInfoDO> getInfo() {
 
       returninfo;
 
    }
 
    publicvoid setInfo(List<BrandContentInfoDO> info) {
 
       this.info = info;
 
    }
 
    publicList<Rate> getRate() {
 
       returnrate;
 
    }
 
    publicvoid setRate(List<Rate> rate) {
 
       this.rate = rate;
 
    }
 
}

经常碰到的问题及解决办法:

  • Exception1:

Unrecognized field “INFO” (Class com.tmall.brand.api.test.model.InfoAndRateMapper), not marked as ignorable

【异常解释】:当反序列化的json对象中有“INFO”,但mapper对象中没有该字段时报错。

【解决办法】:在mapper中增加此变量

  • Exception2:

Unrecognized field “typeString”

【异常解释】:反序列化过程中出现不可识别的成员变量

【解决办法】:增加mapper的属性配置:objectMapper .configure(Feature.FAIL_ON_UNKNOWN_PROPERTIES, false);此参数默认是true,即检查映射关系,当设置为false以后,objectMapper将忽略需解析的json串中“不认识”的字段

例如:mapper对象中有private String name;的对象,当json对象中并没有此变量,那当这个属性值设置为false时,可成功反序列化。

  • Exception3:

org.codehaus.jackson.map.JsonMappingException : Can not instantiate value of type [collection type; class java.util.ArrayList, contains [simple type, class java.lang.Long]] from JSON String; no single-String constructor/factory method (through reference chain: com.tmall.brand.api.test.model.GetInfoAndRateRespModel["result"]->com.tmall.brand.api.test.model.InfoAndRateMapper["INFO"]->com.tmall.brand.service.domain.ratenews.BrandNewsQuery["typeList"])

【异常解释】:jackson无法实例化json对象中变量类型的值

【解决办法】:@JsonIgnoreProperties({ “typeList”})

在mapper类之前加此标签忽略这个字段的反序列化

  • Exception4:

Can not deserialize instance of java.util.ArrayList out of START_OBJECT token

【异常解释】:无法反序列化没有START_OBJECT token的ArrayList

【解决办法】:增加mapper的属性配置:

1
objectMapper .configure(Feature.ACCEPT_SINGLE_VALUE_AS_ARRAY, true);
  • Exception5:

org.codehaus.jackson.map.JsonMappingException : No suitable constructor found for type [simple type, class com.taobao.matrix.snsplugin.common.enumconstants.EnumResultStatus]: can not instantiate from JSON object (need to add/enable type information?)

at [Source: java.io.StringReader@98350a; line: 1, column: 89] (through reference chain: com.tmall.brand.api.test.model.BaseResponseModel["result"]->com.tmall.brand.api.test.model.PublishCommentResultMapper["status"])

【异常解释】:jackson无法实例化json对象,没有找到适合的构造函数。

【解决办法】:jackson在进行反序列号的时候,需要Mapper类是普通的pojo类,并且类中的变量都需要有setXXX的set方法,如果没有set方法或者没有“正确”的set方法,就会报无法实例化JSON对象的异常

例如:

1、json对象的一个变量为

1
privateboolean isSuccess;

自动生成代码后的set方法为:(根据你自己的eclipse 的autogen来决定自动生成的代码模板)

1
2
3
4
5
publicvoid setSuccess( boolean isSuccess) {
 
       this. isSuccess = isSuccess;
 
}

如果是这样,jackson在反序列化的时候就会报此异常,应改为:

1
2
3
4
5
publicvoid setIsSuccess( boolean isSuccess) {
 
             this. isSuccess = isSuccess;
 
}

2、当JSON对象为:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
publicclass XXXStatus extendsXXXResultStatus<XXXStatus> {
 
privatestatic final long serialVersionUID = 2385584133823877965L;
 
publicstatic final Map<Integer, XXXStatus> codeMap = newHashMap<Integer, XXXStatus>();
 
publicstatic final XXXStatus SUCCESS = newXXXStatus(0,"xxx","xxx");
....
 
privateXXXStatus(intcode, Stringmsg, StringfrontMsg) {
    super(code, msg, frontMsg);
    codeMap.put(code,this);
}
 
publicstatic final XXXStatus getRelationStatus(intcode) {
    XXXStatus satus = codeMap.get(code);
    if(satus == null) {
        returnSYSTEM_ERROR;
    }
    returnsatus;
}
}

同样没有set方法,所以这样的JSON对象都需要重新创建一个自定义的mapper对象,用于jackson反序列化

json解析之jackson的更多相关文章

  1. Json解析工具Jackson(使用注解)

    原文http://blog.csdn.net/nomousewch/article/details/8955796 接上一篇文章Json解析工具Jackson(简单应用),jackson在实际应用中给 ...

  2. Java进阶学习:JSON解析利器JackSon

    Java:JSON解析利器JackSon JackSon基础 1.Maven项目引入 <!-- https://mvnrepository.com/artifact/org.codehaus.j ...

  3. Json解析工具Jackson(使用注解)--jackson框架自定义的一些json解析注解

    Json解析工具Jackson(使用注解)--jackson框架自定义的一些json解析注解 @JsonIgnoreProperties 此注解是类注解,作用是json序列化时将Javabean中的一 ...

  4. Json解析工具Jackson(简单应用)

    原文http://blog.csdn.net/nomousewch/article/details/8955796 概述 Jackson库(http://jackson.codehaus.org),是 ...

  5. json解析之jackson ObjectMapper

    Json解析常用的有fastjson和jackson,性能上网上有不少的对比,说是fastjson比较好,今天先整理一下jackson的东西,后面再发一个fastjson的. jackson是spri ...

  6. spring boot2 修改默认json解析器Jackson为fastjson

    0.前言 fastjson是阿里出的,尽管近年fasjson爆出过几次严重漏洞,但是平心而论,fastjson的性能的确很有优势,尤其是大数据量时的性能优势,所以fastjson依然是我们的首选:sp ...

  7. 开源工具-Json 解析器 Jackson 的使用

    Json已经成为当前服务器与 WEB 应用之间数据传输的公认标准.Java 中常见的 Json 类库有 Gson.JSON-lib 和 Jackson 等.相比于其他的解析工具,Jackson 简单易 ...

  8. JSON解析工具——Jackson的简单使用

    什么是Jackson 可以轻松实现Java对象与JSON字符串的转换 准备工作:导包 Jackson的jar all下载地址:http://jackson.codehaus.org/1.7.6/jac ...

  9. 将SpringBoot默认Json解析框架jackson替换成fastjson

    步骤一:引入依赖<dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson< ...

随机推荐

  1. Java学习笔记十:Java的数组以及操作数组

    Java的数组以及操作数组 一:什么是数组: 数组可以理解为是一个巨大的“盒子”,里面可以按顺序存放多个类型相同的数据,比如可以定义 int 型的数组 scores 存储 4 名学生的成绩 数组中的元 ...

  2. ctf题目writeup(2)

    2019.1.29 题目地址: https://www.ichunqiu.com/battalion 1. 点开链接: include "flag.php";$a = @$_REQ ...

  3. PHP中文乱码分类及解决办法大全

    PHP+MYSQL做网站开发通常都会碰到浏览器输出中文字符时乱码,这个问题的原因主要是因为HTML内容编码,PHP文件编码和MySQL数据库编码这三者不一致造成的.下面我们以UTF-8为例简述一下如何 ...

  4. Git 查看远程分支、本地分支、删除本地分支及远程分支

    1. 删除本地分支: git branch -d branchName 2. 删除远程分支: // 方法一:将删除的本地分支推到远程(要删除的远程分支在本地有映射) git push origin : ...

  5. Ubantu修改主机名详细步骤

    使用vmWare创建的Ubantu虚拟主机,默认的主机名均为Ubantu,通过修改主机名的使得分布式集群操作变得方便.具体步骤如下: 1.查看虚拟机的主机名:打开终端,使用 hostname 命令,回 ...

  6. Extjs报错:isField为空或不是对象

             在做Extjs开发的时候,有时候会碰到一个奇怪的问题,就是报错说"isField为空或不是对象",经过调试发现是一个数组,显示的长度是21,但是数组里面的个数只有 ...

  7. How to set pycharm configure for remoting development

    配置pycharm远程连接,点击pycharm的tools,选择deployment选项,选择configuration. 2 点击左侧的加号按钮,新增一个连接,取个名字,根据个人配置选择协议,这里选 ...

  8. js 操作表格行数的删减

    沉溺了好几个月了,自从年假回来就一直在忙换工作的事情: 新环境.新同事,一如既往的工作, 那么闲话不多说,前两天师妹问我要一个类似于添加和删除的demo:闲暇时间我就参照一些代码写了一下, (发现有错 ...

  9. 「日常训练」Phone Numbers (CFR466D2C)

    题意(Codeforces 940C) 给定一字符串,求比它字典序大的字符串.限定其长度,并且只能用原串的字母. 分析 考虑原串长度lorigin与给定的长度lgiven.若给定长度大于原串长度,直接 ...

  10. 第十篇 Python的字符串格式化

    字符串格式化:就是按照你的意愿做一个拼接的过程. 1. 字符串格式化的第一种方式:百分号方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存. %[ ...