fastjson 过滤不需要序列化的属性
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。
Java transient关键字
Java规范原文The transient marker is not fully specified by the JavaLanguageSpecification 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是一种解决办法。
基于JAVA的JSON主要开源项目及其对比
Json开源项目非常多,如org.json、 JSON-Lib、jsontool、Jackson、Gson、SimpleJSON等等,后来专门查看了几种json开源测试数据对比后,决定采用fastjson。展示两组测试数据。首先来看大侠wangym(原博客http://wangym.iteye.com/blog/738933)对Jackson、JSON-Lib、Gson的测试结果
JSON转Bean,5个线程并发,约200字节对象,1千万次转换:
|
Jackson |
JSON-lib |
Gson |
|
|
吞吐量 |
64113.7 |
8067.4 |
13952.8 |
|
总耗时(秒) |
155 |
1238 |
700 |
Bean转JSON,5个线程并发,约200字节对象,1千万次转换:
|
Jackson |
JSON-lib |
Gson |
|
|
吞吐量 |
54802 |
15093.2 |
17308.2 |
|
总耗时(秒) |
181 |
661 |
560 |
显而易见,无论是哪种形式的转换,Jackson > Gson > Json-lib。
Jackson的处理能力甚至高出Json-lib有10倍左右
然后再拿温少的fastjson与JSON-Lib、Simple-JSON、Jackson性能测试对比数据
性能对比
|
测试案例 |
JSON-Lib |
Simple-JSON |
Fastjson |
Jackson |
|
IntArray1000Decode |
3,626 |
1,431 |
563 |
596 |
|
StringArray1000Decode |
2,698 |
2,283 |
677 |
774 |
|
Map100StringDecode |
515 |
597 |
208 |
230 |
功能对比
|
特性 |
JSON-Lib |
Simple-JSON |
Fastjson |
Jackson |
|
序列化支持数组 |
不支持 |
不支持 |
支持 |
支持 |
|
序列化支持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= userService.getUserList();
PrintWriterout = 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定义改成
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>这段伪码:
….
List<User> ls= userService.getUserList();
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.getPropertyFilters().add(filter);
serializer.write(ls);
PrintWriterout = null;
try {
out = getResponse().getWriter();
out.write(sw.toString());
out.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
out.close();
}
…
这样在碰到场景B时就使用第三种办法把remark这个成员给过滤掉,在场景A的情况下不加过滤器即可。
fastjson 过滤不需要序列化的属性的更多相关文章
- fastjson过滤不需要的属性
以下是一个通用的对象转json的方法,使用的fastjson的SimplePropertyPreFilter 对象,个人感觉比使用PropertyPreFilter的匿名内部类形式的过滤器更好用!直接 ...
- .net mvc web api 返回 json 内容,过滤值为null的属性
原文:http://blog.csdn.net/xxj_jing/article/details/49508557 版权声明:本文为博主原创文章,未经博主允许不得转载. .net mvc web ap ...
- C# .net mvc web api 返回 json 内容,过滤值为null的属性
在WebApiConfig.Register 中增加一段 #region 过滤值为null的属性 //json 序列化设置 GlobalConfiguration.Configuration.Form ...
- 用来控制 XML 序列化的属性
通过将下表中的属性应用于类和类成员,可以控制 XmlSerializer 序列化或反序列化该类的实例的方式.若要了解这些属性如何控制 XML 序列化,请参见使用属性控制 XML 序列化. 这些属性还可 ...
- C#序列化json属性名首字母变成小写的解决方案
原文:C#序列化json属性名首字母变成小写的解决方案 //接口返回自动转小写,容易造成前后端不一致,获取不到数据,切换成转驼峰(首字母大写)如Code/Result //在ConfigureServ ...
- .NET中XML序列化和反序列化常用类和用来控制XML序列化的属性总结(XmlSerializer,XmlTypeAttribute,XmlElementAttribute,XmlAttributeAttribute,XmlArrayAttribute...)
序列化和反序列化是指什么? 序列化(seriallization): 将对象转化为便于传输的数据格式, 常见的序列化格式:二进制格式,字节数组,json字符串,xml字符串.反序列化(deserial ...
- [转]FastJSON通过SerializeFilter定制序列化
原文地址:https://github.com/alibaba/fastjson/wiki/SerializeFilter 简介 SerializeFilter是通过编程扩展的方式定制序列化.fast ...
- .net mvc web api 返回 json 内容时过滤值为null的属性
1.响应内容(过滤前) {"msg":"初始化成功!","code":"","success":tr ...
- fastjson 过滤不需要的字段或者只要某些字段
/* * 第一种:在对象响应字段前加注解,这样生成的json也不包含该字段. * @JSONField(serialize=false) * private String name; */ / ...
随机推荐
- main中的argc,argv
那么我们运行程序时,传入的参数,就是这个argc的值:从截图中,我们很清楚的可以看出,argc是传入参数的个数,”传入的参数“加上可执行文件的文件名:
- element-UI 点击一行,背景色变化
代码: @row-click="rowClick" 当某一行被点击时会触发该事件 :row-class-name="tableRowClassName" 可以 ...
- 如何定义一个BUG
一.划分一个bug的等级 bug等级主要分为致命.严重.一般.轻微或者建议四个等级: 1.致命错误:系统无法执行.崩溃或严重资源不足.应用模块无法启动或异常退出.无法测试.造成系统不稳定.价值较高功能 ...
- 【JZOJ3293】【BZOJ4416】【luoguP3989】阶乘字符串
description 给定一个由前n个小写字母组成的串S. 串S是阶乘字符串当且仅当前n个小写字母的全排列(共n!种)都作为S的子序列(可以不连续)出现. 由这个定义出发,可以得到一个简单的枚举法去 ...
- CUDA并行计算 | CUDA算法效率提升关键点概述
文章目录 前言 存取效率 计算效率 性能优化要点 展现足够的并行性 优化内存访问 优化指令执行 前言 CUDA算法的效率总的来说,由存取效率和计算效率两类决定,一个好的CUDA算法必定会让两类效率 ...
- NX二次开发-UFUN拾取向量对话框UF_UI_specify_vector
#include <uf.h> #include <uf_ui.h> UF_initialize(); //拾取向量对话框 ], pnt[]; int mode = UF_UI ...
- NX二次开发-UFUN获取当前显示部件的TAG,UF_PART_ask_display_part
NX9+VS2012 #include <uf.h> #include <uf_modl.h> #include <uf_part.h> UF_initialize ...
- NX二次开发-UFUN删除链表函数UF_MODL_delete_list
NX9+VS2012 #include <uf.h> #include <uf_modl.h> #include <uf_obj.h> #include <u ...
- 秦曾昌人工智能课程---6、Decision Tree Learning
秦曾昌人工智能课程---6.Decision Tree Learning 一.总结 一句话总结: 怎样去构建决策树:比如一维:***|00|***|000|***,|为分割线,每个分割点都是一种情况, ...
- Python 爬虫-抓取小说《盗墓笔记-怒海潜沙》
最近想看盗墓笔记,看了一下网页代码,竟然不是js防爬虫,那就用简单的代码爬下了一节: """ 爬取盗墓笔记小说-七星鲁王宫 """ from ...