记一次fastjson使用过程中的隐藏坑点

现象:
服务A向服务B发请求,B服务http response很慢,10~120秒之间,在请求一段时间后,会把服务A拖垮。
服务A/B均使用了组件C

定位过程:
看了服务B的接口代码,发现并没有很复杂的逻辑,仅仅是一次sql的insert,通过日志观察,耗时主要是在组件C内部的一个map的put过程,put前后均有log,两段log耗时很长,第二段log刚好将map打印出来了。
1、put过程不应该慢,map为带锁的map,但并发请求没有那么大
2、观察第二段log,打印出的map对象有一个很奇怪的对象内容,改对象里的属性在对应的类里找不到,但是确实日志打印了出来
仍然不知道耗时在哪,观察第二段日志的map里奇怪对象,发现对象的奇怪属性的名字和该对象的get方法名相同,而第二段日志里使用了fastjson的toJSONString方法,将map对象转化为JSON String,觉得此处有蹊跷,于是看了下toJSONString的源码,发现该方法将对象序列化的过程中,使用反射将对象的get方法执行了一遍,同时作为对象属性生成到JSONString里,于是才在map里看到了奇怪对象的奇怪属性。而这个get方法本身并不是简单的VO对象方法,而是一个service对象的方法,里面是有数据库和其他逻辑操作的,耗时应该就在这里。
将组件C的这行日志去掉,接口响应均达到了毫秒级。

定位结论:
使用fastjson或类似组件时,内部很可能用反射的方法做一些奇怪的操作,导致外部应用层的性能下降;因此需要养成良好的编码习惯
1、非VO类避免使用getXXX作为函数名称
2、日志中避免直接打印对象,应写toString方法
3、在使用fastjson的toJSONString方法是,可以通过传入SerializerFeature.XXX来控制序列化过程,本例中修改为IgnoreNonFieldGetter、WriteMapNullValue,减少无效的输出
参考

是的 是的
是的 是的
QuoteFieldNames 输出key时是否使用双引号,默认为true
UseSingleQuotes 使用单引号而不是双引号,默认为false
WriteMapNullValue 是否输出值为null的字段,默认为false
WriteEnumUsingToString Enum输出name()或者original,默认为false
UseISO8601DateFormat Date使用ISO8601格式输出,默认为false
WriteNullListAsEmpty List字段如果为null,输出为[],而非null
WriteNullStringAsEmpty 字符类型字段如果为null,输出为”“,而非null
WriteNullNumberAsZero 数值字段如果为null,输出为0,而非null
WriteNullBooleanAsFalse Boolean字段如果为null,输出为false,而非null
SkipTransientField 如果是true,类中的Get方法对应的Field是transient,序列化时将会被忽略。默认为true
SortField 按字段名称排序后输出。默认为false
WriteTabAsSpecial 把\t做转义输出,默认为false 不推荐
PrettyFormat 结果是否格式化,默认为false
WriteClassName 序列化时写入类型信息,默认为false。反序列化是需用到
DisableCircularReferenceDetect 消除对同一对象循环引用的问题,默认为false
WriteSlashAsSpecial 对斜杠’/’进行转义
BrowserCompatible 将中文都会序列化为\uXXXX格式,字节数会多一些,但是能兼容IE 6,默认为false
WriteDateUseDateFormat 全局修改日期格式,默认为false。JSON.DEFFAULT_DATE_FORMAT = “yyyy-MM-dd”;JSON.toJSONString(obj, SerializerFeature.WriteDateUseDateFormat);
DisableCheckSpecialChar 一个对象的字符串属性中如果有特殊字符如双引号,将会在转成json时带有反斜杠转移符。如果不需要转义,可以使用这个属性。默认为false
NotWriteRootClassName 含义
BeanToArray 将对象转为array输出
WriteNonStringKeyAsString 含义
NotWriteDefaultValue 含义
BrowserSecure 含义
IgnoreNonFieldGetter 对于在类field中没有声明的属性不做转换
WriteEnumUsingName 含义
发布了12 篇原创文章 · 获赞 0 · 访问量 569

猜你喜欢

转载自blog.csdn.net/weixin_42305433/article/details/90729660