天天看点

FastJson 我大意了,我没有闪

前言

我没有闪

正文

最近在做项目,在对象判空的时候,根据业务需求,写了个自动判空的方法 isEmpty()。

代码示例:

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;
import lombok.Data;
import lombok.experimental.Accessors;

/**
 * @Author: JCccc
 * @Description:
 * @Date: 2020/11/23 10:13
 */
@Data
@Accessors(chain = true)
public class Programmer {

    private Integer id;
    private String name;
    private Integer age;
    private String jobNum;
    private String imgUrl;
    private String cardId;
    private String phone;

    public boolean isEmpty() {
        return null == this.jobNum;
    }


}      

这一切都看似正常,但是在写完业务逻辑后,自测地时候发现返回json格式数据时,多了个字段 empty。

直觉告诉我,这个empty肯定跟我自己写的判空方法有关。但是一下子还不知道为啥。

FastJson 我大意了,我没有闪

问题复现,就是转成JSON数据返回时,发现一个key 为 empty ,值为false  :

代码示例:

public static void main(String[] args) {
        Programmer programmer=new Programmer();
        programmer.setId(1);
        programmer.setName("JCccc");
        programmer.setAge(12);
        programmer.setJobNum("10010");
        programmer.setImgUrl(null);
        programmer.setCardId("xxxxxxxxxxxxx");
        programmer.setPhone("136xxxxxxxx");
        


        JSONObject jsonObject= (JSONObject) JSON.toJSON(programmer);
        System.out.println(jsonObject.toJSONString());
    }      

 输出结果:

FastJson 我大意了,我没有闪

这个 empty 怎么出现的呢 ?

一下子有点蒙圈,但是咱们也是武林中人,走一波正常排查流程。

1.根据直觉,肯定跟这个方法有关。

于是乎,我索性把方法注释掉,测试,发现empty字段消失了。

问题锁定,就是这个方法。

2.那么根据方法的编译时规则,辨别方法的身份特征,是 方法名 和 参数。

而我写的判空方法,是没有传递参数的,所以,进一步确定, 方法名问题。

于是乎我把方法名随便改了个名字,testA(),果然一测试,发现empty字段消失了。

这一下子,思路清晰了,好像想到了一些东西,没错,就是阿里巴巴的开发手册好像有这么一个注意项,如图:

FastJson 我大意了,我没有闪

没错,正是因为这个is开头的命名导致的!!!

通过debug也是可以看到的,有兴趣的可以debug直接看这个方法:

第一个debug点:

FastJson 我大意了,我没有闪

第二个debug点:

FastJson 我大意了,我没有闪

第三个debug点:

FastJson 我大意了,我没有闪

第四个debug点:(其实到了这里,针对方法的话,fastJson还是给了机会的,只需要使用@JSONField(serialize=false) 注解就好了)

FastJson 我大意了,我没有闪

好吧,就先这样吧。

我已经改成verifyEmpty()了,毕竟都是前人踩坑总结出来的,就注意好了。

(本人已测试,不止fastjson,还有其他序列化如果使用is开头的变量,也会有影响。 )