fastjson 过滤不需要序列化的属性

前端之家收集整理的这篇文章主要介绍了fastjson 过滤不需要序列化的属性前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

JavaJSON技术框架选型与实例@H_301_3@

@H_301_3@

JSON

JSON英文全称为JavaScriptObject Natation,采用key:value键值对的方式存贮数据,与xml格式相比,JSON是一种轻量级的数据交换格式;不要被Javascript这个单词迷惑,实际上JSON只是一种数据格式,与具体语言并无关系。JSON已被广泛应用于业界,比如目前Nosql数据库存贮大都采用key:value存贮结构,以Mongo为例,其脚本语法甚至直接使用Javascript;在数据传输时,采用JSON格式也被广泛应用,大部分开放API都开放JSON模式的数据输出;在ajax请求数据时,json格式也被广泛推荐。json更多信息的可以查看json官方网站http://json.org。@H_301_3@

Java transient关键字

JAVA规范原文The transient marker is not fully specified by the Java LanguageSpecification but is used in object serialization to mark member variables thatshould 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是一种解决办法。@H_301_3@

基于JAVA的JSON主要开源项目及其对比

Json开源项目非常多,如org.json、JSON-Lib、jsontool、Jackson、Gson、SimpleJSON等等,后来专门查看了几种json开源测试数据对比后,决定采用fastjson。展示两组测试数据。首先来看大侠wangym(原博客http://wangym.iteye.com/blog/738933)对Jackson、JSON-Lib、Gson的测试结果@H_301_3@

JSON转Bean,5个线程并发,约200字节对象,1千万次转换:@H_301_3@

Jackson@H_301_3@

JSON-lib@H_301_3@

Gson@H_301_3@

吞吐量@H_301_3@

64113.7@H_301_3@

8067.4@H_301_3@

13952.8@H_301_3@

总耗时(秒)@H_301_3@

155@H_301_3@

1238@H_301_3@

700@H_301_3@

Bean转JSON,5个线程并发,约200字节对象,1千万次转换:@H_301_3@

Jackson@H_301_3@

JSON-lib@H_301_3@

Gson@H_301_3@

吞吐量@H_301_3@

54802@H_301_3@

15093.2@H_301_3@

17308.2@H_301_3@

总耗时(秒)@H_301_3@

181@H_301_3@

661@H_301_3@

560@H_301_3@

显而易见,无论是哪种形式的转换,Jackson > Gson > Json-lib@H_301_3@

Jackson的处理能力甚至高出Json-lib10倍左右@H_301_3@

@H_301_3@

然后再拿温少的fastjson与JSON-Lib、Simple-JSON、Jackson性能测试对比数据@H_301_3@

性能对比@H_301_3@

测试案例@H_301_3@

JSON-Lib@H_301_3@

Simple-JSON@H_301_3@

Fastjson@H_301_3@

Jackson@H_301_3@

IntArray1000Decode@H_301_3@

3,626@H_301_3@

1,431@H_301_3@

563@H_301_3@

596@H_301_3@

StringArray1000Decode@H_301_3@

2,698@H_301_3@

301_3@

677@H_301_3@

774@H_301_3@

Map100StringDecode@H_301_3@

515@H_301_3@

597@H_301_3@

208@H_301_3@

230@H_301_3@

功能对比@H_301_3@

特性@H_301_3@

JSON-Lib@H_301_3@

Simple-JSON@H_301_3@

Fastjson@H_301_3@

Jackson@H_301_3@

序列化支持数组@H_301_3@

不支持@H_301_3@

支持@H_301_3@

支持@H_301_3@

序列化支持Enum@H_301_3@

支持JavaBean@H_301_3@

不直接支持@H_301_3@

支持@H_301_3@

可以看到Fastjson性能方面,超越目前的所有java json proccesor包括jackson@H_301_3@

@H_301_3@

FastJson应用实例

1、利用Jquery ajax请求fastjson数据来显示用户列表例子实现

//定义一个User PO对象@H_301_3@

publicclassUserimplementsSerializable{@H_301_3@

@H_301_3@

privatestaticfinallongserialVersionUID= 1738399846398814044L;@H_301_3@

privateStringuserid;@H_301_3@

username;@H_301_3@

//注意这里使用了Refrence及Lazyloading相关的引用@H_301_3@

@Refrence@H_301_3@

privateUserDetail userDeatil;@H_301_3@

publicString getUserid() {@H_301_3@

return }@H_301_3@

voidsetUserid(String userid) {@H_301_3@

this.userid= userid;@H_301_3@

publicString getUsername() {@H_301_3@

voidsetUsername(String username) {@H_301_3@

this. username = username;@H_301_3@

publicUserDetail getUserDetail() {@H_301_3@

returnuserDetail;@H_301_3@

voidsetUserDetail (UserDetail userDetail) {@H_301_3@

this. userDetail = userDetail;@H_301_3@

}@H_301_3@

}@H_301_3@

UserDetail POclassUserDetailserialVersionUID= 1738399846398814045L;@H_301_3@

address;@H_301_3@

publicString getAddress() {@H_301_3@

address;@H_301_3@

voidsetAddress (Stringaddress) {@H_301_3@

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

….@H_301_3@

List<User> ls=userService.getUserList();@H_301_3@

PrintWriterout =null;@H_301_3@

try{@H_301_3@

out = getResponse().getWriter();@H_301_3@

out.write(JSON.toJSONString(ls));@H_301_3@

out.flush();@H_301_3@

}catch(IOException e) {@H_301_3@

e.printStackTrace();@H_301_3@

finally{@H_301_3@

out.close();@H_301_3@

…@H_301_3@

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

$.ajax({@H_301_3@

type:"GET",@H_301_3@

url:"/user/getuserlist",//假设这是你配置后的action地址@H_301_3@

dataType:"json",68)"> cache:false,@H_301_3@

success:function(users){@H_301_3@

var html=””;@H_301_3@

if(users.length>0){@H_301_3@

for(variinusers){@H_301_3@

html=html+”username:”+users[i]+username+” address:”+users[i].userDetail.address;@H_301_3@

alert(html);@H_301_3@

});@H_301_3@

@H_301_3@

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

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

第一种办法:@H_301_3@

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

private transientUserDetailuserDeatil;@H_301_3@

第二种办法:@H_301_3@

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

@JSONField(serialize=false)@H_301_3@

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

第三种办法:@H_301_3@

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

userService.getUserList();@H_301_3@

PropertyFilter filter = new PropertyFilter() {@H_301_3@

publicbooleanapply(Object source,String name,Object value) {@H_301_3@

if("remark".equals(name)) {@H_301_3@

return true;@H_301_3@

returnfalse;@H_301_3@

};@H_301_3@

SerializeWriter sw = new SerializeWriter();@H_301_3@

JSONSerializer serializer = new JSONSerializer(sw);@H_301_3@

serializer.getPropertyFilters().add(filter);@H_301_3@

serializer.write(ls);@H_301_3@

sw.toString());@H_301_3@

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

更多fastjson信息可以查看http://code.alibabatech.com/wiki/display/FastJSON/Home@H_301_3@ 原文链接:https://www.f2er.com/json/290252.html

猜你在找的Json相关文章