遭遇fastjson的一个坑

前端之家收集整理的这篇文章主要介绍了遭遇fastjson的一个坑前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

最近在开发过程中踩到到了fastjson的一个坑,写出来跟大家分享一下,同时也留个记录。
情况是这样子的,我们中间件是采用fastjson来进行序列化的,我们自己在代码中定义一个Response类用于封装两个系统之间交换的数据,代码简化版大致如下:

  1. public class Response<T> {
  2.  
  3. private int status;//用于标志调用状态
  4.  
  5. private T data;//实际数据
  6.  
  7. public int getStatus() {
  8. return status;
  9. }
  10.  
  11. public void setStatus(int status) {
  12. this.status = status;
  13. }
  14.  
  15. public T getData() {
  16. return data;
  17. }
  18.  
  19. public void setData(T data) {
  20. this.data = data;
  21. }
  22.  
  23. public boolean isSuccess() {
  24. return data != null ? true : false;
  25. }
  26.  
  27. public void setSuccess(T data){
  28. this.data = data;
  29. this.status = SUCCESS;
  30. //do something success
  31. }
  32.  
  33. @Override
  34. public String toString() {
  35. return "Response{" +
  36. "status=" + status +
  37. ",data=" + data +
  38. '}';
  39. }
  40. }

Response类中封装一个泛类型data成员为实际需要的数据,定义的datagetset方法,定义的一个isSuccess()方法来判定系统是否成功给client端用。定义了一个setSuccess()方法给server端用。

如下实例化一个Response对象:

  1. Response<String> response = new Response<>();
  2. response.setData("test");
  3. response.setStatus(0);
  4. String json = JSON.toJSONString(response);//fastjson序列化
  5. System.out.println(json);//打印出序列化json
  6. response = JSON.parSEObject(json,Response.class);//fastjson序列化反java对象
  7. System.out.println(response);//打印出反序列化java对象

预计json序列化结果为:

  1. {"data":"test","status":0}

而实际用“,输出的json确实如下:

  1. {"data":"test","status":0,"success":true}

发现了没,序列化的json中多了一个success,这跟设计的不太一样啊!!这可不是我要的结果….多了个对实际程序影响还没有那么大,更坑爹的还在后面的反序列化,输出结果如下:

  1. Response{status=0,data=true}

而不是:

  1. Response{status=0,data="test"}

data中的数据被篡改了,类型都不对了有木有…这将导致很严重的后果!

这主要是Response不是一个标准的POJO类,isSuccess()setSuccess()刚好被fastjson多解析为successgetset方法,然后setSuccess()在反序列化时把序列化结果中的success值覆盖掉了data,多个条件碰到了一起,造成了悲剧…
其实解决办法很简单,使用标准的POJO或者修改一下isSuccess()setSuccess()方法名即可。 因此建议大家在使用类似于fastjson这种工具进行序列化时要注意改问题,避免这种悲剧的重演。

猜你在找的Json相关文章