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 条评论
登录 后参与评论

相关文章

来自专栏编程之旅

数据结构——无权图的路径问题(C++和java实现)

好像又是接近半个月没有更新,这半个月忙着结婚的各项事情,本来预计的学习任务也拖拖拉拉,进度缓慢。吐槽一句,拍婚纱照真的是最非常非常累的一件事情,不想再有下次了。

1192
来自专栏Java进阶

为什么会要序列化

44610
来自专栏PHP在线

PHP程序员容易忽略的几点精华

1、变量、数组的应用技巧   (1)很多人用得不多的数组函数。foreach、list、each。分别举几个例子,应该就能知道了。例:   $dat...

37110
来自专栏ImportSource

快来了解JDK10中引入的全新JIT编译器:Graal

在(JDK10要来了:下一代 Java 有哪些新特性?)文中,我们提到jdk10中包含有一个实验性质的编译器(compiler)。它的名字叫做:Graal。这是...

56911
来自专栏我有一个梦想

UE4中的单映射:TMap容器

一、TMap<T>是么 TMap<T>是UE4中的一种关联容器,每个键都关联着一个值,形成了单映射关系。因此你可以通过键名来快速查找到值。此外,单映射要求每...

2369
来自专栏北京马哥教育

让你的 Python 代码优雅又地道

学Python最简单的方法是什么?推荐阅读:Python开发工程师成长魔法 译序 如果说优雅也有缺点的话,那就是你需要艰巨的工作才能得到它,需要良好的教育才能欣...

40310
来自专栏个人随笔

Java工具类 通过ResultSet对象返回对应的实体List集合

4215
来自专栏风中追风

分布式基础__为什么会要序列化

对 java对象的传输 是通过网络的 。然后网络都是通过字节传输的 所以需要用序列化的方式 将java对象转化为 字节,然后网络那头的接受端,拿到收到的字节 再...

3517
来自专栏九彩拼盘的叨叨叨

如何给函数取个合适的名字

Quora 和 Ubuntu Forums thread 上的 4500 个程序员对上面的问题进行投票。49%的程序员认为给函数,变量等命名是最难的任务。

782
来自专栏IMWeb前端团队

简洁的javascript编码(一)--变量、函数

本文作者:IMWeb jaychen 原文出处:IMWeb社区 未经同意,禁止转载 ? 一、变量 使用语义化的变量名称 Bad cons...

2539

扫码关注云+社区

领取腾讯云代金券