本文出处:http://blog.csdn.net/chaijunkun/article/details/8257209,转载请注明。因为本人不按期会整理相关博文,会对相应内容做出完善。所以强烈建议在原始出处查看此文。
在年初的时候,我曾经写过一篇文章介绍非关系型数据库MongoDB和Jackson JSON框架相结合实现接口敏捷开发的文章(http://blog.csdn.net/chaijunkun/article/details/7263804),被可爱的CSDN小编推到了首页。在此本人对小编表示感谢。事隔10个月,随着手头一些项目的进行,对Jackson JSON框架用得愈来愈多。以为有必要再写点什么补充出来。做为和广大同仁的一个经验的分享。
咱们都知道,Jackson JSON以高速、方便和灵活著称。以前的文章中介绍过使用注解的形式来规定如何将一个对象序列化成JSON的方法,以及如何将一个JSON数据反序列化到一个对象上。可是美中不足的一点就是对于中文的处理。固然我说的美中不足是在默认状况下,Jackson JSON不会将中文等非ASCII字符转换为\uFFFF这样的形式来显示。也就是说默认状况下会显示为{"name":"张三"}而不是{"name":"\u5F20\u4E09"}。那么为何有这样的需求呢?在HTTP协议中,咱们能够指定数据头部分的内容编码。如:“GBK”、“UTF-8”等等。若是你设置正确了,那么OK,前者所表示的数据您能够正确处理。然而若是设置错误,对于中文字符将会产生乱码。两套应用系统对接,有可能两边使用的默认编码不一样,若是一方修改默认编码将会对应用形成不可预知的后果。所以若能以长远的眼光开发,那么不管您设置成什么编码方式,都不会使数据产生乱码。由于,这里用到了万国编码——Unicode。
好的,问题出来了,咱们如何解决呢?使其经过实验,Jackson JSON其实在默认设置下已经具有了对Unicode编码的JSON数据进行解析。所欠缺的就是在序列化对象时缺乏相应的步骤。好在Jackson JSON框架容许咱们自定义序列化方法。那么咱们就来写一个序列化类:java
- package net.csdn.blog.chaijunkun.util;
-
- import java.io.IOException;
-
- import org.codehaus.jackson.JsonGenerationException;
- import org.codehaus.jackson.JsonGenerator;
- import org.codehaus.jackson.JsonProcessingException;
- import org.codehaus.jackson.impl.JsonWriteContext;
- import org.codehaus.jackson.map.JsonSerializer;
- import org.codehaus.jackson.map.SerializerProvider;
- import org.codehaus.jackson.util.CharTypes;
-
- public class StringUnicodeSerializer extends JsonSerializer<String> {
-
- private final char[] HEX_CHARS = "0123456789ABCDEF".toCharArray();
- private final int[] ESCAPE_CODES = CharTypes.get7BitOutputEscapes();
-
- private void writeUnicodeEscape(JsonGenerator gen, char c) throws IOException {
- gen.writeRaw('\\');
- gen.writeRaw('u');
- gen.writeRaw(HEX_CHARS[(c >> 12) & 0xF]);
- gen.writeRaw(HEX_CHARS[(c >> 8) & 0xF]);
- gen.writeRaw(HEX_CHARS[(c >> 4) & 0xF]);
- gen.writeRaw(HEX_CHARS[c & 0xF]);
- }
-
- private void writeShortEscape(JsonGenerator gen, char c) throws IOException {
- gen.writeRaw('\\');
- gen.writeRaw(c);
- }
-
- @Override
- public void serialize(String str, JsonGenerator gen,
- SerializerProvider provider) throws IOException,
- JsonProcessingException {
- int status = ((JsonWriteContext) gen.getOutputContext()).writeValue();
- switch (status) {
- case JsonWriteContext.STATUS_OK_AFTER_COLON:
- gen.writeRaw(':');
- break;
- case JsonWriteContext.STATUS_OK_AFTER_COMMA:
- gen.writeRaw(',');
- break;
- case JsonWriteContext.STATUS_EXPECT_NAME:
- throw new JsonGenerationException("Can not write string value here");
- }
- gen.writeRaw('"');
- for (char c : str.toCharArray()) {
- if (c >= 0x80){
- writeUnicodeEscape(gen, c);
- }else {
-
- int code = (c < ESCAPE_CODES.length ? ESCAPE_CODES[c] : 0);
- if (code == 0){
- gen.writeRaw(c);
- }else if (code < 0){
- writeUnicodeEscape(gen, (char) (-code - 1));
- }else {
- writeShortEscape(gen, (char) code);
- }
- }
- }
- gen.writeRaw('"');
- }
-
- }
这个序列化类将要对应用中全部使用Jackson JSON的地方全都用一种方法来处理字符串类型。光有了方法还不行,还要对它进行注册。让Jackson JSON在序列化对象的时候使用刚刚定义好的方法:
- if (objectMapper== null){
- objectMapper= new ObjectMapper();
-
- objectMapper.configure(SerializationConfig.Feature.FAIL_ON_EMPTY_BEANS, false);
-
- CustomSerializerFactory serializerFactory= new CustomSerializerFactory();
- serializerFactory.addSpecificMapping(String.class, new StringUnicodeSerializer());
- objectMapper.setSerializerFactory(serializerFactory);
-
- }
2014年5月13日补充:最近被问到不少次关于单例模式的实现。上面的写法真的很不安全,没有加锁,也没有对objectMapper进行volatile修饰(即所谓的“双检索”货“双重检查”),所以最简单的可靠的方法应该使用“枚举单例法”。数据库
2014年11月21日补充:因为Jackson 2的版本变化,CustomSerializerFactory已经被去掉了,通过实验,可使用这种方式代替:apache
- if (objectMapper== null){
- objectMapper= new ObjectMapper();
-
- objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
-
- SimpleModule module = new SimpleModule();
- module.addSerializer(String.class, new StringUnicodeSerializer());
- objectMapper.registerModule(module);
-
- objectMapper.setSerializationInclusion(Include.NON_NULL);
-
- }
接下来咱们来作一个测试用的对象,验证咱们的代码:json
- package net.csdn.blog.chaijunkun.json;
-
- import java.util.Date;
-
- import net.csdn.blog.chaijunkun.util.DateDeserializer;
- import net.csdn.blog.chaijunkun.util.DateSerializer;
- import net.csdn.blog.chaijunkun.util.DateTimeDeserializer;
- import net.csdn.blog.chaijunkun.util.DateTimeSerializer;
-
- import org.codehaus.jackson.annotate.JsonPropertyOrder;
- import org.codehaus.jackson.map.annotate.JsonDeserialize;
- import org.codehaus.jackson.map.annotate.JsonSerialize;
-
- @JsonPropertyOrder(alphabetic= false)
- public class DemoObj {
-
- private Integer sid;
-
- private String stuName;
-
- private Boolean sex;
-
- @JsonSerialize(using= DateSerializer.class)
- @JsonDeserialize(using= DateDeserializer.class)
- private Date birthday;
-
- @JsonSerialize(using= DateTimeSerializer.class)
- @JsonDeserialize(using= DateTimeDeserializer.class)
- private Date logTime;
-
-
-
- }
从代码上能够看出,咱们并无对String类型的属性强制指定用何种序列与反序列方法。而后咱们来构造测试用例:
- package net.csdn.blog.chaijunkun.test;
-
- import java.text.SimpleDateFormat;
- import java.util.Calendar;
- import java.util.Date;
-
- import net.csdn.blog.chaijunkun.json.DemoObj;
- import net.csdn.blog.chaijunkun.util.JSONUtil;
-
- import org.apache.log4j.Logger;
-
- public class JSONTest {
-
- private static Logger logger= Logger.getLogger(JSONTest.class);
-
- private static String json= "{\"sid\":2,\"stuName\":\"\u6C5F\u5357Style\",\"sex\":true,\"birthday\":\"2012-07-15\",\"logTime\":\"2012-12-04 19:22:36\"}";
-
- public static void main(String[] args) {
- DemoObj objSrc= new DemoObj();
- objSrc.setSid(1);
- objSrc.setStuName("鸟叔");
- objSrc.setSex(true);
- Calendar calendar= Calendar.getInstance();
- calendar.set(1977, Calendar.DECEMBER, 31, 0, 0, 0);
- objSrc.setBirthday(calendar.getTime());
- objSrc.setLogTime(new Date());
- logger.info(String.format("转换为JSON后的数据:%s", JSONUtil.toJSON(objSrc)));
- DemoObj objDes= JSONUtil.fromJSON(json, DemoObj.class);
- if(objDes==null){
- logger.info("反序列化失败");
- }else{
- logger.info("反序列化成功");
- SimpleDateFormat sdf= new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
- logger.info(String.format("标识:%d", objDes.getSid()));
- logger.info(String.format("姓名:%s", objDes.getStuName()));
- logger.info(String.format("性别:%s", objDes.getSex()==true?"男":"女"));
- logger.info(String.format("生日:%s", sdf.format(objDes.getBirthday())));
- logger.info(String.format("登陆日期:%s", sdf.format(objDes.getLogTime())));
- }
- }
-
- }
看一下输出:安全
- 转换为JSON后的数据:{"sid":1,"stuName":"\u9E1F\u53D4","sex":true,"birthday":"1977-12-31","logTime":"2012-12-04 19:31:57"}
- 反序列化成功
- 标识:2
- 姓名:江南Style
- 性别:男
- 生日:2012-07-15 00:00:00
- 登陆日期:2012-12-04 19:22:36
咱们看到,已经成功将中文字符显示成为了Unicode编码的数据。一样,咱们以前构造的Unicode编码的数据,在不通过任何修改的状况下成功显示出来了。
细心的朋友也许观察到了,在测试用的对象定义代码中,针对一样Date类型的属性“birthday”和“logTime”,咱们指定了不一样的序列化与反序列化方法。让咱们来看烂这两个有什么不一样:app
- package net.csdn.blog.chaijunkun.util;
-
- import java.io.IOException;
- import java.text.SimpleDateFormat;
- import java.util.Date;
-
- import org.codehaus.jackson.JsonGenerator;
- import org.codehaus.jackson.JsonProcessingException;
- import org.codehaus.jackson.map.JsonSerializer;
- import org.codehaus.jackson.map.SerializerProvider;
-
- public class DateTimeSerializer extends JsonSerializer<Date> {
-
- @Override
- public void serialize(Date date, JsonGenerator gen, SerializerProvider provider)
- throws IOException, JsonProcessingException {
- SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
- String formattedDate= sdf.format(date);
- gen.writeString(formattedDate);
- }
-
- }
- package net.csdn.blog.chaijunkun.util;
-
- import java.io.IOException;
- import java.text.SimpleDateFormat;
- import java.util.Calendar;
- import java.util.Date;
-
- import org.codehaus.jackson.JsonParser;
- import org.codehaus.jackson.JsonProcessingException;
- import org.codehaus.jackson.map.DeserializationContext;
- import org.codehaus.jackson.map.JsonDeserializer;
-
- public class DateTimeDeserializer extends JsonDeserializer<Date> {
-
- @Override
- public Date deserialize(JsonParser parser, DeserializationContext context)
- throws IOException, JsonProcessingException {
- String dateFormat= "yyyy-MM-dd HH:mm:ss";
- SimpleDateFormat sdf= new SimpleDateFormat(dateFormat);
- try{
- String fieldData= parser.getText();
- return sdf.parse(fieldData);
- }catch (Exception e) {
- Calendar ca= Calendar.getInstance();
- ca.set(1970, Calendar.JANUARY, 1, 0, 0, 0);
- return ca.getTime();
- }
- }
- }
- package net.csdn.blog.chaijunkun.util;
-
- import java.io.IOException;
- import java.text.SimpleDateFormat;
- import java.util.Date;
-
- import org.codehaus.jackson.JsonGenerator;
- import org.codehaus.jackson.JsonProcessingException;
- import org.codehaus.jackson.map.JsonSerializer;
- import org.codehaus.jackson.map.SerializerProvider;
-
- public class DateSerializer extends JsonSerializer<Date> {
-
- @Override
- public void serialize(Date date, JsonGenerator gen, SerializerProvider provider)
- throws IOException, JsonProcessingException {
- SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
- String formattedDate= sdf.format(date);
- gen.writeString(formattedDate);
- }
-
- }
- package net.csdn.blog.chaijunkun.util;
-
- import java.io.IOException;
- import java.text.SimpleDateFormat;
- import java.util.Calendar;
- import java.util.Date;
-
- import org.codehaus.jackson.JsonParser;
- import org.codehaus.jackson.JsonProcessingException;
- import org.codehaus.jackson.map.DeserializationContext;
- import org.codehaus.jackson.map.JsonDeserializer;
-
- public class DateDeserializer extends JsonDeserializer<Date> {
-
- @Override
- public Date deserialize(JsonParser parser, DeserializationContext context)
- throws IOException, JsonProcessingException {
- String dateFormat= "yyyy-MM-dd";
- SimpleDateFormat sdf= new SimpleDateFormat(dateFormat);
- try{
- String fieldData= parser.getText();
- return sdf.parse(fieldData);
- }catch (Exception e) {
- Calendar ca= Calendar.getInstance();
- ca.set(1970, Calendar.JANUARY, 1, 0, 0, 0);
- return ca.getTime();
- }
- }
- }
从代码咱们能够看出,DateTimeSerializer和DateTimeDeserializer比DateSerializer和DateDeserializer细粒度更加高,加入了具体时间的属性。这在应用开发中是很常见的,生日信息咱们每每知道年月日就能够了,而登录时间每每须要得比较详细。从实例中咱们能够知道,即使是同一类型,经过制定不一样的序列与反序列方法,能够灵活地获得咱们想要的数据形态。以上测试用例已经打包,上传到了个人资源。欢迎你们下载,共同窗习。下载地址:http://download.csdn.net/detail/chaijunkun/4846394
2012年12月17日补充:框架
最近有一个需求,须要在序列化与反序列化对象的时候对数据进行修改,当发现数据源值为空时须要让生成的JSON显示改字段为“游客”。但是我不管如何指定序列化器与反序列化器都无效。程序根本走不到指定的代码中去。后来我得出结论,Jackson JSON在反序列化对象的时候,若JSON数据中对应属性为null,则不会走自定义的反序列化器;一样地,当你设置对象的某个属性值为null时,在将其序列化成JSON时,也不会走自定义的序列化器。所以如有相似的需求,请在序列化与反序列化以前经过硬代码形式判断和修改,千万不要什么事都期望着序列化器与反序列化器。ide
参考资料:来源于国外网站的一篇介绍如何转码的文章,原文有点错误。我将其改正了,并加入了一些中文注释:http://wiki.fasterxml.com/JacksonSampleQuoteChars学习