如何解決Fastjson日期轉(zhuǎn)換引起的生產(chǎn)事故

如何解決Fastjson日期轉(zhuǎn)換引起的生產(chǎn)事故,針對(duì)這個(gè)問題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問題的小伙伴找到更簡單易行的方法。

創(chuàng)新互聯(lián)是由多位在大型網(wǎng)絡(luò)公司、廣告設(shè)計(jì)公司的優(yōu)秀設(shè)計(jì)人員和策劃人員組成的一個(gè)具有豐富經(jīng)驗(yàn)的團(tuán)隊(duì),其中包括網(wǎng)站策劃、網(wǎng)頁美工、網(wǎng)站程序員、網(wǎng)頁設(shè)計(jì)師、平面廣告設(shè)計(jì)師、網(wǎng)絡(luò)營銷人員及形象策劃。承接:網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作、網(wǎng)站改版、網(wǎng)頁設(shè)計(jì)制作、網(wǎng)站建設(shè)與維護(hù)、網(wǎng)絡(luò)推廣、數(shù)據(jù)庫開發(fā),以高性價(jià)比制作企業(yè)網(wǎng)站、行業(yè)門戶平臺(tái)等全方位的服務(wù)。

生產(chǎn)環(huán)境: Linux,Java ,JDK 1.7,F(xiàn)astjson1.2.72(alibaba)

2020年9月15日,fastjson從1.2.51升級(jí)到1.2.72版本后,在調(diào)用JSON.toJSONString(this)或者new JSONObject().toJSON().toString()方法時(shí),如果日期類型是java.sql.Date,并且為yyyy-mm-dd格式,則在調(diào)用以上方法后輸出的是yyyy-mm-dd格式的字符串。

而在1.2.51版本中,這種情況下輸出的格式是時(shí)間戳; 測試代碼如下:

AccountPayFlow accountPayFlow = new AccountPayFlow();
Date date = DateUtil.parse("2020-11-06");
Date sqlDate = new java.sql.Date(date.getTime());
accountPayFlow.setGiftExpireDate(sqlDate);
System.out.println(accountPayFlow.toString());
System.out.println(new JSONObject().toJSON(accountPayFlow).toString());  

//輸出結(jié)果  fastjson 1.2.72版本 
{"giftExpireDate":"2020-11-06"}
{"giftExpireDate":"2020-11-06"}

// 輸出結(jié)果 fastjson 1.2.51版本
{"giftExpireDate":1604592000000}
{"giftExpireDate":1604592000000}

問題分析

首先,java.sql.Date繼承自java.util.Date。

fastjson1.2.51版本并沒有明確區(qū)分java.sql.Date和java.util.Date,而是籠統(tǒng)的判斷對(duì)象如果是Date類型(包括Date本身和其子類),就直接強(qiáng)制轉(zhuǎn)換為java.util.Date,后續(xù)的所有操作都是基于java.util.Date進(jìn)行的。

在后續(xù)的操作中,對(duì)于Date類的的數(shù)據(jù),會(huì)通過getTime()獲取到Long類型的時(shí)間戳,然后調(diào)用out.writeLong()將Long類型轉(zhuǎn)換為String 。

因此 在此版本下的日期格式通過fastjson轉(zhuǎn)換后是一個(gè)時(shí)間戳。

截取部分fastjson源碼如下

 //com.alibaba.fastjson.serializer.DateCodec.write()  versions:1.2.51
public void write(JSONSerializer serializer, Object object, Object fieldName, Type fieldType, int features) throws IOException {
        SerializeWriter out = serializer.out;
        if (object == null) {
            out.writeNull();
        } else {
            Date date;
            if (object instanceof Date) {
                date = (Date)object;
            } else {
                date = TypeUtils.castToDate(object);
            }
                // ....省略部分代碼
                long time = date.getTime();
                // ....省略部分代碼
                out.writeLong(time);
            }
        }
    }

fastjson1.2.72版本對(duì)java.sql.Date進(jìn)行的特殊判斷,如果日期是java.sql.Date,并且格式是yyyy-mm-dd格式,則直接調(diào)用java.sql.Date的toString()方法,而java.sql.Date的toString()方法會(huì)返回yyyy-mm-dd格式的日期字符串,導(dǎo)致最終輸出的結(jié)果是yyyy-mm-dd格式的字符串。

具體源碼如下:

//com.alibaba.fastjson.serializer.DateCodec.write()  versions:1.2.72
public void write(JSONSerializer serializer, Object object, Object fieldName, Type fieldType, int features) throws IOException {
        SerializeWriter out = serializer.out;

        if (object == null) {
            out.writeNull();
            return;
        }
        // ....省略部分代碼 
        //1.2.72版本增加了對(duì)java.sql.Date類型的特殊處理
        Class<?> clazz = object.getClass();
        if (clazz == java.sql.Date.class) {
            long millis = ((java.sql.Date) object).getTime();
            TimeZone timeZone = serializer.timeZone;
            int offset = timeZone.getOffset(millis);
            //
            if ((millis + offset) % (24 * 1000 * 3600) == 0
                    && !SerializerFeature.isEnabled(out.features, features, SerializerFeature.WriteClassName)) {
                // 此處直接調(diào)用java.sql.Date的toString()方法,該方法
                out.writeString(object.toString());
                return;
            }
        }
    
    // 后續(xù)的代碼邏輯與1.2.51基本一致
      Date date;
            if (object instanceof Date) {
                date = (Date)object;
            } else {
                date = TypeUtils.castToDate(object);
            }
                // ....省略部分代碼
                long time = date.getTime();
                // ....省略部分代碼
                out.writeLong(time);
            }
    }

java.sql.Date的toString()方法 只返回yyyy-mm-dd

    /**
     * Formats a date in the date escape format yyyy-mm-dd.
     * <P>
     * @return a String in yyyy-mm-dd format
     */
    @SuppressWarnings("deprecation")
    public String toString () {
        int year = super.getYear() + 1900;
        int month = super.getMonth() + 1;
        int day = super.getDate();

        char buf[] = "2000-00-00".toCharArray();
        buf[0] = Character.forDigit(year/1000,10);
        buf[1] = Character.forDigit((year/100)%10,10);
        buf[2] = Character.forDigit((year/10)%10,10);
        buf[3] = Character.forDigit(year%10,10);
        buf[5] = Character.forDigit(month/10,10);
        buf[6] = Character.forDigit(month%10,10);
        buf[8] = Character.forDigit(day/10,10);
        buf[9] = Character.forDigit(day%10,10);

        return new String(buf);
    }

fastjson各版本對(duì)日期格式的處理方式梳理

在fastjson1.2.56(包括1.2.56)之前,對(duì)于日期類型的處理方式一律是轉(zhuǎn)換為java.util.Date,然后進(jìn)行處理,在這種情況下,無論java.util.Date還是java.sql.Date,無論日期格式是yyyy-mm-dd 還是 yyyy-mm-dd HH:mm:ss 或者其他日期格式,都是按照同一套邏輯進(jìn)行處理的。

從fastjson1.2.57版本開始,對(duì)于日期的格式而言,增加了對(duì)java.sql.Date的特殊處理,具體已在上文中有詳細(xì)描述。

從1.2.57版本之后一直到當(dāng)前最新版本1.2.73,針對(duì)java.sql.Date類型的特殊處理邏輯,雖局部有調(diào)整,但上文中描述的判斷邏輯一直保持不變。

關(guān)于如何解決Fastjson日期轉(zhuǎn)換引起的生產(chǎn)事故問題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識(shí)。

文章題目:如何解決Fastjson日期轉(zhuǎn)換引起的生產(chǎn)事故
文章分享:http://bm7419.com/article38/psdopp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供做網(wǎng)站、網(wǎng)站設(shè)計(jì)、品牌網(wǎng)站設(shè)計(jì)、網(wǎng)站營銷App設(shè)計(jì)、靜態(tài)網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

微信小程序開發(fā)