Java JSON技术框架选型与实例

JSON

JSON英文全称为JavaScript Object Natation,采用key:value键值对的方式存贮数据,与xml格式相比,JSON是一种轻量级的数据交换格式;不要被Javascript这个单词迷惑,实际上JSON只是一种数据格式,与具体语言并无关系。J

</pre>SON已被广泛应用于业界,比如目前NoSQL数据库存贮大都采用key:value存贮结构,以Mongo为例,其脚本语法甚至直接使用Javascript;在数据传输时,采用JSON格式也被广泛应用,大部分开放API都开放JSON模式的数据输出;在ajax请求数据时,json格式也被广泛推荐。json更多信息的可以查看json官方网站http://json.org。</p><h1>Java transient关键字</h1><p>   JAVA规范原文The transient marker is not fully specified by the Java Language Specification but is used in object serialization to mark member variables that should not be serialized.为了方便存贮和网络传输,java有系列化对象机制,transient可用来指定当前不想被系列化的成员对象。举个例子说明transient的应用,在Mongo+Morphia开源项目下,如果对Java PO的成员指定transient,那么该成员数据将不会被存入Mongo数据库。另外一种应用场景就是这里要讲到的JSON,如果JAVA PO使用了Refrence(Mongo的Refrence)或者LazyLoading(可以理解成Hibernate LazyLoading概念),那么大部分的开源JAVA JSON相关项目,会自动加载这些Refrence、LazyLoading对象,如果PO形成相互引用,那就会形成死循环,即使没有形成死循环,大量不必要的数据被输出到客户端对资源的浪费也不容小觑。加上transient是一种解决办法。</p><h1>基于JAVA的JSON主要开源项目及其对比</h1><p>Json开源项目非常多,如org.json、 JSON-Lib、jsontool、Jackson、Gson、SimpleJSON等等,后来专门查看了几种json开源测试数据对比后,决定采用fastjson。展示两组测试数据。首先来看大侠wangym(原博客<a target=_blank href="http://wangym.iteye.com/blog/738933" style="color:rgb(16, 138, 198);">http://wangym.iteye.com/blog/738933</a>)对Jackson、JSON-Lib、Gson的测试结果</p><p><strong>JSON转Bean,5个线程并发,约200字节对象,1千万次转换:</strong></p><table cellpadding="0" width="593" border="1" style="color:rgb(0, 0, 0);"><tbody><tr><td> </td><td><p>Jackson</p></td><td><p>JSON-lib</p></td><td><p>Gson</p></td></tr><tr><td><p>吞吐量</p></td><td><p>64113.7</p></td><td><p>8067.4</p></td><td><p>13952.8</p></td></tr><tr><td><p>总耗时(秒)</p></td><td><p>155</p></td><td><p>1238</p></td><td><p>700</p></td></tr></tbody></table><p> <strong>Bean转JSON,5个线程并发,约200字节对象,1千万次转换:</strong></p><table cellpadding="0" width="593" border="1" style="color:rgb(0, 0, 0);"><tbody><tr><td> </td><td><p>Jackson</p></td><td><p>JSON-lib</p></td><td><p>Gson</p></td></tr><tr><td><p>吞吐量</p></td><td><p>54802</p></td><td><p>15093.2</p></td><td><p>17308.2</p></td></tr><tr><td><p>总耗时(秒)</p></td><td><p>181</p></td><td><p>661</p></td><td><p>560</p></td></tr></tbody></table><p><strong><span style="color:red;">显而易见,无论是哪种形式的转换,</span>Jackson > Gson > Json-lib<span style="color:red;">。</span></strong></p><p>     <strong>Jackson<span style="color:red;">的处理能力甚至高出</span>Json-lib<span style="color:red;">有</span>10<span style="color:red;">倍左右</span></strong></p><p> </p><p>然后再拿温少的fastjson与JSON-Lib、Simple-JSON、Jackson性能测试对比数据</p><p><a target=_blank name="Tutorial-%E6%80%A7%E8%83%BD%E5%AF%B9%E6%" style="color:rgb(16, 138, 198);"></a><strong>性能对比</strong></p><table cellpadding="0" cellspacing="0" border="0" style="color:rgb(0, 0, 0);"><tbody><tr><td><p><strong>测试案例</strong></p></td><td><p><strong>JSON-Lib</strong></p></td><td><p><strong>Simple-JSON</strong></p></td><td><p><strong>Fastjson</strong></p></td><td><p><strong>Jackson</strong></p></td></tr><tr><td><p><span style="color:rgb(51, 51, 51);">IntArray1000Decode</span></p></td><td><p><span style="color:rgb(51, 51, 51);">3,626</span></p></td><td><p><span style="color:rgb(51, 51, 51);">1,431</span></p></td><td><p><span style="color:rgb(51, 51, 51);">563</span></p></td><td><p><span style="color:rgb(51, 51, 51);">596</span></p></td></tr><tr><td><p><span style="color:rgb(51, 51, 51);">StringArray1000Decode</span></p></td><td><p><span style="color:rgb(51, 51, 51);">2,698</span></p></td><td><p><span style="color:rgb(51, 51, 51);">2,283</span></p></td><td><p><span style="color:rgb(51, 51, 51);">677</span></p></td><td><p><span style="color:rgb(51, 51, 51);">774</span></p></td></tr><tr><td><p><span style="color:rgb(51, 51, 51);">Map100StringDecode</span></p></td><td><p><span style="color:rgb(51, 51, 51);">515</span></p></td><td><p><span style="color:rgb(51, 51, 51);">597</span></p></td><td><p><span style="color:rgb(51, 51, 51);">208</span></p></td><td><p><span style="color:rgb(51, 51, 51);">230</span></p></td></tr></tbody></table><p><a target=_blank name="Tutorial-%E5%8A%9F%E8%83%BD%E5%AF%B9%E6%" style="color:rgb(16, 138, 198);"></a><strong>功能对比</strong></p><table cellpadding="0" cellspacing="0" border="0" style="color:rgb(0, 0, 0);"><tbody><tr><td><p><strong>特性</strong></p></td><td><p><strong>JSON-Lib</strong></p></td><td><p><strong>Simple-JSON</strong></p></td><td><p><strong>Fastjson</strong></p></td><td><p><strong>Jackson</strong></p></td></tr><tr><td><p>序列化支持数组</p></td><td><p>不支持</p></td><td><p>不支持</p></td><td><p>支持</p></td><td><p>支持</p></td></tr><tr><td><p>序列化支持Enum</p></td><td><p>不支持</p></td><td><p>不支持</p></td><td><p>支持</p></td><td><p>支持</p></td></tr><tr><td><p>支持JavaBean</p></td><td><p>不直接支持</p></td><td><p>不直接支持</p></td><td><p>支持</p></td><td><p>支持</p></td></tr></tbody></table><p> </p><p><strong><span style="color:red;">可以看到</span>Fastjson<span style="color:red;">在性能方面,超越目前的所有</span>java json proccesor<span style="color:red;">,包括</span>jackson<span style="color:red;">。</span></strong></p><p><strong> </strong></p><h1>FastJson应用实例</h1><h2>1、利用Jquery ajax请求fastjson数据来显示用户列表例子实现</h2><p><strong><span courier="" new="" color:="" 7f0055="" style="font-size: 10pt;">//</span><span style="color:rgb(127, 0, 85);">定义一个</span><span courier="" new="" color:="" 7f0055="" style="font-size: 10pt;">User PO</span><span style="color:rgb(127, 0, 85);">对象</span></strong></p><pre name="code" class="java">public class User implements Serializable {  
     
    private static final long serialVersionUID = 1738399846398814044L;  
     
    private String userid;  
    private String username;  
    //注意这里使用了Refrence及Lazyloading相关的引用  
@Refrence  
    private UserDetail userDeatil;  
    public String getUserid() {  
       return userid;  
    }  
    public void setUserid(String userid) {  
       this.userid = userid;  
    }  
    public String getUsername() {  
       return username;  
    }  
    public void setUsername(String username) {  
       this. username = username;  
    }  
public UserDetail getUserDetail() {  
       return userDetail;  
    }  
    public void setUserDetail (UserDetail userDetail) {  
       this. userDetail = userDetail;  
    }  
}


//定义一个UserDetail PO对象

public class UserDetail implements Serializable {  
     
    private static final long serialVersionUID = 1738399846398814045L;  
     
    private String address;  
     
public String getAddress() {  
       return address;  
    }  
    public void setAddress (String address) {  
       this. address = address;  
    }  
}

编写Action,输出List<User>,这里使用伪码

….  
List<User> ls= userService.getUserList();  
PrintWriter out = null;  
       try {  
           out = getResponse().getWriter();  
           out.write(JSON.toJSONString(ls));  
           out.flush();  
       } catch (IOException e) {  
           e.printStackTrace();  
       } finally {  
           out.close();  
       }  
…  

编写jquery ajax请求打出用户列表

$.ajax({  
       type:"GET",  
       url:"/user/getuserlist", //假设这是你配置后的action地址  
       dataType:"json",  
       cache:false,  
       success: function(users){  
            var html=””;  
            if(users.length>0){  
           for(var i in users){  
               html=html+”username:”+users[i]+username+” address:”+users[i].userDetail.address;  
           }  
          alert(html);  
}  
});  

2、如何解决Refrence及LazyLoading引起的死循环问题?

从上述例子可以看到fastjson会正确取出userDetail下的address数据,实际上所有的json开源项目都支持这种关联取出。但有时候我们并不需要userDetail下的数据,如果自动加载一堆无关的数据,甚至产生死循环,怎么解决呢?

第一种办法:

前面已经讲过,加上transient关键字,如给User PO的UserDetail定义改成

Java代码   收藏代码
  1. <strong>private transient  UserDetail userDeatil;  
  2.  </strong>  
 

第二种办法:

第一种办法是通用的办法,使用其他json开源项目,也可以达到效果,在FastJson下还可以使用@JSONField(serialize=false)

Java代码   收藏代码
  1. @JSONField(serialize=false)  
  2. private transient  UserDetail userDeatil;  
 

当然JSONField还有其他参数可以指定,以实现成员定制序列化,一般情况下,如果我们确定成员可以为非序列化,首先建议使用transient。但有时候指定了transient会引起其他问题,假如User对象下有长字段remark,如果给remark指定了transient,那么在比如使用Mongo数据库情况下,会导致页面提交的remark数据不能被保存到数据库,其他没有加transient关键字的字段能正常保存。这时就可以使用@JSONField来解决问题。

 

第三种办法:

   假如有更进一步的优化,比如场景A的时候需要系列化remark,而在场景B的时候又不需要系列化,那就使用fastjson定制过滤器,fastjson可以按name、property、value三种过滤,以property例,重写获取List<user>这段伪码:

Java代码   收藏代码
  1. ….  
  2. List<User> ls= userService.getUserList();  
  3. PropertyFilter filter = new PropertyFilter() {  
  4.     public boolean apply(Object source, String name, Object value) {  
  5.         if("remark ".equals(name)) {  
  6.             return true;  
  7.         }  
  8.         return false;  
  9.     }  
  10. };  
  11.  SerializeWriter sw = new SerializeWriter();  
  12. JSONSerializer serializer = new JSONSerializer(sw);  
  13. serializer.getPropertyFilters().add(filter);  
  14. serializer.write(ls);  
  15. PrintWriter out = null;  
  16.        try {  
  17.            out = getResponse().getWriter();  
  18.            out.write(sw.toString());  
  19.            out.flush();  
  20.        } catch (IOException e) {  
  21.            e.printStackTrace();  
  22.        } finally {  
  23.            out.close();  
  24.        }  
  25. …  
 

这样在碰到场景B时就使用第三种办法把remark这个成员给过滤掉,在场景A的情况下不加过滤器即可。

更多fastjson信息可以查看http://code.alibabatech.com/wiki/display/FastJSON/Home

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值