http://wujiu.iteye.com/blog/2244537
我們在使用Spring MVC 是都有這樣的經歷,@ResponseBody返回一個JSON格式字符串到前端,或者使用@RequestBody獲取前端REST提交的對象,那么Spring MVC 中是如何處理的呢?大體的流程如下:
在Spring Web中使用 jackson處理,具體實現類是MappingJackson2HttpMessageConverter.?具體的JSON系列化和反序列化都是有jackson實現的。下面主要介紹一下jackson的使用。
對日期進行JSON序列化:
Java代碼
/**
*
*?@author?zhangwei
*?@since?2015年9月18日?下午4:08:08
*?@version?v?0.1
*/
publicclassContainDateObject?{
privateDate?startTime;
privateDate?endTime;
publicDate?getStartTime()?{
returnstartTime;
}
publicvoidsetStartTime(Date?startTime)?{
this.startTime?=?startTime;
}
publicDate?getEndTime()?{
returnendTime;
}
publicvoidsetEndTime(Date?endTime)?{
this.endTime?=?endTime;
}
}
Java代碼
@Test
publicvoidtestDate()throwsJsonProcessingException?{
ContainDateObject?cdo?=newContainDateObject();
cdo.setStartTime(newDate());
cdo.setEndTime(newDate());
System.out.println(objectMapper.writeValueAsString(cdo));
}
結果是:
Java代碼
{"startTime":1442563815734,"endTime":1442563815734}
這樣的結果有的時候是無法接受的,我們有的時候需要一個yyyy-MM-dd HH:mm:ss這樣格式的字符串改如何處理呢?這個時候我們就需要制定一個自定義的序列化類;
Java代碼
/**
*?Desc:自定義日期類型序列化類
*
*?@author?zhangwei
*?@since?2015年9月18日?下午4:12:51
*?@version?v?0.1
*/
publicclassDateJsonSerializerextendsJsonSerializer?{
/**
*?@see?com.fasterxml.jackson.databind.JsonSerializer#serialize(java.lang.Object,
*??????com.fasterxml.jackson.core.JsonGenerator,
*??????com.fasterxml.jackson.databind.SerializerProvider)
*/
@Override
publicvoidserialize(Date?date,?JsonGenerator?generator,?SerializerProvider?provider)
throwsIOException,?JsonProcessingException?{
ObjectMapper?om?=newObjectMapper();
SimpleDateFormat?sdf?=newSimpleDateFormat("yyyy-MM-dd?HH:mm:ss");
om.writeValue(generator,?sdf.format(date));
}
}
在需要按照指定格式進行序列化的日期屬性上添加注解
Java代碼
@JsonSerialize(using?=?DateJsonSerializer.class)
privateDate?startTime;
@JsonSerialize(using?=?DateJsonSerializer.class)
privateDate?endTime;
測試的結果是:
Java代碼
{"startTime":"2015-09-18?16:23:23","endTime":"2015-09-18?16:23:23"}
這樣的結果就比較符合預期了
反過來,使用這個JSON轉換為一個對象時的結果怎樣呢?
Java代碼
@Test
publicvoidtestDate()throwsException?{
ContainDateObject?cdo?=newContainDateObject();
cdo.setStartTime(newDate());
cdo.setEndTime(newDate());
String?serializerResult?=?objectMapper.writeValueAsString(cdo);
cdo?=?objectMapper.readValue(serializerResult,?ContainDateObject.class);
}
結果卻是:
Java代碼
com.fasterxml.jackson.databind.exc.InvalidFormatException:?Can?not?construct?instance?of?java.util.Date?from?String?value'2015-09-18?16:26:58':?not?a?valid?representation?(error:?Failed?to?parse?Date?value'2015-09-18?16:26:58':?Can?not?parse?date"2015-09-18?16:26:58":?not?compatible?with?any?of?standard?forms?("yyyy-MM-dd'T'HH:mm:ss.SSSZ","yyyy-MM-dd'T'HH:mm:ss.SSS'Z'","EEE,?dd?MMM?yyyy?HH:mm:ss?zzz","yyyy-MM-dd"))
at?[Source:?{"startTime":"2015-09-18?16:26:58","endTime":"2015-09-18?16:26:58"};?line:1,?column:2]?(through?reference?chain:?com.david.demo.jackson.ContainDateObject["startTime"])
at?com.fasterxml.jackson.databind.exc.InvalidFormatException.from(InvalidFormatException.java:55)
at?com.fasterxml.jackson.databind.DeserializationContext.weirdStringException(DeserializationContext.java:883)
at?com.fasterxml.jackson.databind.deser.std.StdDeserializer._parseDate(StdDeserializer.java:750)
at?com.fasterxml.jackson.databind.deser.std.DateDeserializers$DateBasedDeserializer._parseDate(DateDeserializers.java:176)
at?com.fasterxml.jackson.databind.deser.std.DateDeserializers$DateDeserializer.deserialize(DateDeserializers.java:262)
at?com.fasterxml.jackson.databind.deser.std.DateDeserializers$DateDeserializer.deserialize(DateDeserializers.java:246)
at?com.fasterxml.jackson.databind.deser.SettableBeanProperty.deserialize(SettableBeanProperty.java:523)
at?com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:95)
at?com.fasterxml.jackson.databind.deser.impl.BeanPropertyMap.findDeserializeAndSet(BeanPropertyMap.java:285)
at?com.fasterxml.jackson.databind.deser.BeanDeserializer.vanillaDeserialize(BeanDeserializer.java:248)
at?com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:136)
at?com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3562)
at?com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2578)
反序列化失敗了,這又改如何處理呢?我們同樣指定反序列化類
Java代碼
/**
*?Desc:日期類型的反序列化類
*
*?@author?zhangwei
*?@since?2015年9月18日?下午4:31:02
*?@version?v?0.1
*/
publicclassDateJsonDeserializerextendsJsonDeserializer?{
/**
*?@see?com.fasterxml.jackson.databind.JsonDeserializer#deserialize(com.fasterxml.jackson.core.JsonParser,
*??????com.fasterxml.jackson.databind.DeserializationContext)
*/
@Override
publicDate?deserialize(JsonParser?parser,?DeserializationContext?context)
throwsIOException,?JsonProcessingException?{
try{
SimpleDateFormat?sdf?=newSimpleDateFormat("yyyy-MM-dd?HH:mm:ss");
returnsdf.parse(parser.getValueAsString());
}catch(Exception?e)?{
e.printStackTrace();
}
returnnull;
}
}
Java代碼
@JsonSerialize(using?=?DateJsonSerializer.class)
@JsonDeserialize(using?=?DateJsonDeserializer.class)
privateDate?startTime;
@JsonSerialize(using?=?DateJsonSerializer.class)
@JsonDeserialize(using?=?DateJsonDeserializer.class)
privateDate?endTime;
這樣反序列化問題也可以很好地解決了。
我們再想對象中添加一個個人隱私信息字段 phone;
Java代碼
@JsonSerialize(using?=?DateJsonSerializer.class)
@JsonDeserialize(using?=?DateJsonDeserializer.class)
privateDate?startTime;
@JsonSerialize(using?=?DateJsonSerializer.class)
@JsonDeserialize(using?=?DateJsonDeserializer.class)
privateDate?endTime;
privateString?phone;
Java代碼
@Test
publicvoidtestDate()throwsException?{
ContainDateObject?cdo?=newContainDateObject();
cdo.setStartTime(newDate());
cdo.setEndTime(newDate());
cdo.setPhone("13919309243");
String?serializerResult?=?objectMapper.writeValueAsString(cdo);
System.out.println(serializerResult);
objectMapper.readValue(serializerResult,?ContainDateObject.class);
}
這樣序列化的結果是:
Java代碼
{"startTime":"2015-09-18?16:37:34","endTime":"2015-09-18?16:37:34","phone":"13919309243"}
phone是一個個人隱私信息,如果不想被序列化改如何處理?只需要在該屬性上添加一個注解
Java代碼
@JsonIgnore
privateString?phone;
再次序列化的結果是:
Java代碼
{"startTime":"2015-09-18?16:43:15","endTime":"2015-09-18?16:43:15"}