fastjson过滤不需要序列化的属性
JSON
Java transient关键字
规范原⽂The transient marker is not fully specified by the 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(可以理解成 LazyLoading概念),那么⼤部分的开源JAVA JSON相关项⽬,会⾃动加载这些Refrence、LazyLoading对象,如果PO形成相互引⽤,那就会形成死循环,即使没有形成死循环,⼤量不必要的数据被输出到客户端对资源的浪费也不容⼩觑。加上transient是⼀种解决办法。
基于JAVA的JSON主要开源项⽬及其对⽐
JSON转Bean,5个线程并发,约200字节对象,1千万次转换:
Jackson JSON-lib Gson
吞吐量64113.78067.413952.8
总耗时(秒)1551238700
Bean转JSON,5个线程并发,约200字节对象,1千万次转换:
Jackson JSON-lib Gson
吞吐量5480215093.217308.2
总耗时(秒)181661560
显⽽易见,⽆论是哪种形式的转换,Jackson > Gson > Json-lib。
Jackson的处理能⼒甚⾄⾼出Json-lib有10倍左右
然后再拿温少的fastjson与JSON-Lib、Simple-JSON、Jackson性能测试对⽐数据
性能对⽐
测试案例JSON-LibSimple-JSONFastjsonJackson
IntArray1000Decode3,6261,431563596
StringArray1000Decode2,6982,283677774
Map100StringDecode515597208230
功能对⽐
特性JSON-Lib Simple-JSONFastjsonJackson
序列化⽀持数组不⽀持不⽀持⽀持⽀持
序列化⽀持Enum不⽀持不⽀持⽀持⽀持
⽀持JavaBean不直接⽀持不直接⽀持⽀持⽀持
可以看到Fastjson在性能⽅⾯,超越⽬前的所有java json proccesor,包括jackson。
FastJson应⽤实例
1、利⽤Jquery ajax请求fastjson数据来显⽰⽤户列表例⼦实现
/
/定义⼀个User PO对象
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= UserList();
PrintWriterout = null;
try {
out = getResponse().getWriter();
out.JSONString(ls));
out.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
out.close();
}
…
编写 ajax请求打出⽤户列表
$.ajax({
type:"GET",
", //假设这是你配置后的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定义改成
private transient UserDetailuserDeatil;
第⼆种办法:
第⼀种办法是通⽤的办法,使⽤其他json开源项⽬,也可以达到效果,在FastJson下还可以使⽤@JSONField(serialize=false)
@JSONField(serialize=false)
private transient UserDetailuserDeatil;
当然JSONField还有其他参数可以指定,以实现成员定制序列化,⼀般情况下,如果我们确定成员可以为⾮序列化,⾸先建议使⽤transient。但有时候指定了transient会引起其他问题,假如User对象下有长字段remark,如果给remark指定了transient,那么在⽐如使⽤Mongo数据库情况下,会导致页⾯提交的remark数据不能被保存到数据库,其他没有加transient关键字的字段能正常保存。这时就可以使⽤@JSONField来解决问题。
第三种办法:
假如有更进⼀步的优化,⽐如场景A的时候需要系列化remark,⽽在场景B的时候⼜不需要系列化,那就使⽤fastjson定制过滤器,fastjson 可以按name、property、value三种过滤,以property例,重写获取List<user>这段伪码:
….
fastjson怎么用List<User> ls= UserList();
PropertyFilter filter = new PropertyFilter() {
publicboolean apply(Object source, String name, Object value) {
if("remark".equals(name)) {
return true;
}
returnfalse;
}
};
SerializeWriter sw = new SerializeWriter();
JSONSerializer serializer = new JSONSerializer(sw);
serializer.write(ls);
PrintWriterout = null;
try {
out = getResponse().getWriter();
out.String());
out.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
out.close();
}
…
这样在碰到场景B时就使⽤第三种办法把remark这个成员给过滤掉,在场景A的情况下不加过滤器即可。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论