java:关于json解析工具选型(JSON-java,json-lib,gson,fastjson)

没有最好的工具,只有最适合的. 最近的项目需要对java 对象和json之间的序列化和反序列化,更准确的说是java bean对象和json之间的转换,使用哪个工具进行转换,让我犹豫了不少时间。 json工具有很多 http://json.org 中列出不少,我接触过的有JSON-java,json-lib,google-gson,还有阿里巴巴的fastjson

JSON-java

我最早用过的是最简单的JSON-java(json.org官方提供的代码),是大约两年前了,所以这次再用到json的时候,首先想到就是它。 JSON-java代码非常少,也很简洁易懂,对于json入门者真的非常适合阅读理解。 但功能也最简陋,没有提供方便的json->java bean对象的反序列化能力,前两天尝试想修改代码增加反序列化功能,但折腾一天发现修改的工作量实在太大,得不偿失,果断放弃了。

json-lib

对于json-lib,看了它的文档,功能上能满足我的需要,但它至少有4个依赖库,有些臃肿,这是我很在意思的。 而且根据网上的评测,速度也很慢。所以放弃了。

google-gson

google出品的东西质量都不会差,gson拥有非常快的序列化反序列化速度,使用起来也非常方便灵活,网上有不少资料介绍它的用法。 但是目前gson是基于字段序列化的(fields-based),而不是基于getter/setter方法来决定哪些字段需要被序列化(properties),也就是就说对于一个对象中的成员(field),不论它是private还是public,也不论是否有getter/setter方法,都会被序列化. 参见gson官网提供的例子 Object-Examples

However, there are good arguments to support properties as well. We intend to enhance Gson in a latter version to support properties as an alternate mapping for indicating Json fields. For now, Gson is fields-based.
@摘自gson设计文档 GsonDesignDocument

参见: Why does GSON use fields and not getters/setters? GsonDesignDocument

gson这种设计在有的应用场景下非常有用,但对于我的需求而言,这并不合适。 我只希望将拥有getter/setter方法的成员序列化,使用gson会将其他不需要被序列化的private成员也输出到json字符串中,这无疑是多余的,会让json体积更大。

fastjson

阿里巴巴出品的fastjson是唯一有中文说明的json工具,看起来非常方便,也是目前号称java语言中最快的json库。支持getter/setter方法序列化,也支持对public的成员(field)序列化。这样就很灵活了,正好满足我的需求。 下面是测试代码 :

package net.gdface.facelog;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;

import org.junit.Test;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature;

public class TestFastjson {
    public interface Person<T>{

    }
    public static class User implements Person<String>{

        private Long   id;
        private String name;


        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

    }
    public static  class Group {

        private Long       id;
        private String     name;
        private List<User> users = new ArrayList<User>();
        // public成员 不需要getter/setter方法直接序列化
        public Date create_date=new Date();
        public java.sql.Date sqldate=new java.sql.Date(birthdate.getTime());
        public byte[] sampleArray=new byte[]{22,33,3,2,3,1,5,-1};
        // 没有getter/setter方法的privated成员不会被序列化
        private String privString="private string";
        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public List<User> getUsers() {
            return users;
        }

        public void setUsers(List<User> users) {
            this.users = users;
        }

            public void addUser(User user) {
                users.add(user);
            }
    }
    @Test
    public void test() {
        Group group = new Group();
        group.setId(0L);
        group.setName("admin");

        User guestUser = new User();
        guestUser.setId(2L);
        guestUser.setName("guest");

        User rootUser = new User();
        rootUser.setId(3L);
        rootUser.setName("root");

        group.addUser(guestUser);
        group.addUser(rootUser);
        // 对日期使用ISO-8601日期格式
        String jsonString = JSON.toJSONString(group,SerializerFeature.UseISO8601DateFormat);
        // 输出序列化结果
        System.out.println(jsonString);
        // 反序列化
        Group decodeGroup = JSON.parseObject(jsonString, Group.class);
        //输出反序列化结果
        System.out.println(JSON.toJSONString(decodeGroup,SerializerFeature.UseISO8601DateFormat));
    }
}

输出结果

{"array":"FiEDAgMBBf8=","date":"2017-09-07T11:42:13.306+08:00","id":0,"name":"admin","sqldate":"2017-09-07T11:42:13.306+08:00","users":[{"id":2,"name":"guest"},{"id":3,"name":"root"}]}
{"array":"FiEDAgMBBf8=","date":"2017-09-07T11:42:13.306+08:00","id":0,"name":"admin","sqldate":"2017-09-07T11:42:13.306+08:00","users":[{"id":2,"name":"guest"},{"id":3,"name":"root"}]}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏求索之路

Android数据层架构的实现 下篇

接上篇:Android数据层架构的实现 上篇 4.外观模式实现数据处理引擎框架暴露出来的api 我们在使用各种开源框架的时候,大多数时候都不会对框架内部...

2885
来自专栏Android 研究

OkHttp源码解析(十) OKHTTP中连接与请求及总结

主要看下ConnectInterceptor()方法,里面代码已经很简单了,受限了通过streamAllocation的newStream方法获取一个流(Htt...

1023
来自专栏大内老A

如何在ASP.NET Core应用中实现与第三方IoC/DI框架的整合?

我们知道整个ASP.NET Core建立在以ServiceCollection/ServiceProvider为核心的DI框架上,它甚至提供了扩展点使我们可以与...

1785
来自专栏余林丰

多个构造器参数使用构建器

标题一眼看过去可能不是很明白要讲什么,先来看看下面一段代码。 1 package example; 2 3 /** 4 * 重叠构造器 5 * ...

1808
来自专栏大内老A

WCF中关于可靠会话的BUG!!

对WCF的可靠会话编程有一定了解的人应该知道,我们可以使用 DeliveryRequirementsAttribute 可以指示WCF确认绑定提供服务或客户端实...

18210
来自专栏大内老A

WCF技术剖析之五:利用ASP.NET兼容模式创建支持会话(Session)的WCF服务

在《基于IIS的WCF服务寄宿(Hosting)实现揭秘》中,我们谈到在采用基于IIS(或者说基于ASP.NET)的WCF服务寄宿中,具有两种截然不同的运行模式...

1779
来自专栏帅小子的日常

使用redis做缓存

3997
来自专栏流媒体

OkHttp源码深入解读

目前在HTTP协议请求库中,OKHttp应当是非常火的,使用也非常的简单。网上有很多文章写了关于OkHttp的特点以及使用,而本章主要带领大家阅读OkHttp的...

1011
来自专栏python开发教学

restful规范

          restful 规范(10) 什么是接口? - URL - 约束 # 约束继承(实现)了他的类中必须含有IFoo中的...

963
来自专栏大内老A

WCF技术剖析之三十:一个很有用的WCF调用编程技巧[上篇]

在进行基于会话信道的WCF服务调用中,由于受到并发信道数量的限制,我们需要及时的关闭信道;当遇到某些异常,我们需要强行中止(Abort)信道,相关的原理,可以参...

27410

扫码关注云+社区