溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何分析fastjson處理下劃線和駝峰問(wèn)題的方法和源碼

發(fā)布時(shí)間:2021-11-17 16:34:06 來(lái)源:億速云 閱讀:529 作者:柒染 欄目:軟件技術(shù)

如何分析fastjson處理下劃線和駝峰問(wèn)題的方法和源碼,針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。

一. 前言

在開發(fā)過(guò)程中經(jīng)常遇到j(luò)son解析和生成的問(wèn)題,所以用自己也一直用fastjson來(lái)實(shí)現(xiàn)這個(gè)功能。

但是,最近遇到一個(gè)問(wèn)題: 

  1. json字符串里面的數(shù)據(jù)很多都是"_"下劃線的比如,op_id。

  2. 而在java里面,很多都是駝峰的寫法,如opId

那這兩種可以匹配然后解析嗎?

二. http請(qǐng)求的解決方法

http請(qǐng)求有個(gè)@JsonProperty的注解,但是這個(gè)注解,fastjson不識(shí)別;不過(guò)fastjson支持JSONField注解,如下:

@JSONField(name="sta")
private String status;

三. 智能匹配

fastjson提供了智能匹配的規(guī)則,下面寫法會(huì)自動(dòng)映射

op_id->opid->ipId

也就是說(shuō)就算json字符串是'op_id',那java變量也可以用opid或者opId,然后也可以獲取相應(yīng)的數(shù)據(jù)。

如下:

public class Runme {
	static int ONE_DAY_SECONDS = 24 * 60 * 60 * 1000;
	public static void main(String[] args) {
		String json = "{\"op-id\":1000}";
		Mo mo = JSON.parseObject(json, Mo.class);
		
		System.out.println(mo.getOpId());
	}
	
	public static class Mo {
		private String opId;

		public String getOpId() {
			return opId;
		}

		public void setOpId(String opId) {
			this.opId = opId;
		}
	}
}

四. 原理分析

那fastjson是怎么做到的呢?

看了下源代碼 

https://github.com/alibaba/fastjson

發(fā)現(xiàn)它的邏輯如下:

文件:src/main/java/com/alibaba/fastjson/parser/deserializer/JavaBeanDeserializer.java

方法:  smartMatch(String key, int[] setFlags)

public FieldDeserializer smartMatch(String key, int[] setFlags) {
        if (key == null) {
            return null;
        }
        
        FieldDeserializer fieldDeserializer = getFieldDeserializer(key, setFlags);

        if (fieldDeserializer == null) {
            long smartKeyHash = TypeUtils.fnv1a_64_lower(key);
            if (this.smartMatchHashArray == null) {
                long[] hashArray = new long[sortedFieldDeserializers.length];
                for (int i = 0; i < sortedFieldDeserializers.length; i++) {
                    hashArray[i] = TypeUtils.fnv1a_64_lower(sortedFieldDeserializers[i].fieldInfo.name);
                }
                Arrays.sort(hashArray);
                this.smartMatchHashArray = hashArray;
            }

            // smartMatchHashArrayMapping
            int pos = Arrays.binarySearch(smartMatchHashArray, smartKeyHash);
            boolean is = false;
            if (pos < 0 && (is = key.startsWith("is"))) {
                smartKeyHash = TypeUtils.fnv1a_64_lower(key.substring(2));
                pos = Arrays.binarySearch(smartMatchHashArray, smartKeyHash);
            }

            if (pos >= 0) {
                if (smartMatchHashArrayMapping == null) {
                    short[] mapping = new short[smartMatchHashArray.length];
                    Arrays.fill(mapping, (short) -1);
                    for (int i = 0; i < sortedFieldDeserializers.length; i++) {
                        int p = Arrays.binarySearch(smartMatchHashArray
                                , TypeUtils.fnv1a_64_lower(sortedFieldDeserializers[i].fieldInfo.name));
                        if (p >= 0) {
                            mapping[p] = (short) i;
                        }
                    }
                    smartMatchHashArrayMapping = mapping;
                }

                int deserIndex = smartMatchHashArrayMapping[pos];
                if (deserIndex != -1) {
                    if (!isSetFlag(deserIndex, setFlags)) {
                        fieldDeserializer = sortedFieldDeserializers[deserIndex];
                    }
                }
            }

            if (fieldDeserializer != null) {
                FieldInfo fieldInfo = fieldDeserializer.fieldInfo;
                if ((fieldInfo.parserFeatures & Feature.DisableFieldSmartMatch.mask) != 0) {
                    return null;
                }

                Class fieldClass = fieldInfo.fieldClass;
                if (is && (fieldClass != boolean.class && fieldClass != Boolean.class)) {
                    fieldDeserializer = null;
                }
            }
        }


        return fieldDeserializer;
    }

它里面有個(gè)重要的地方就是調(diào)用TypeUtils.fnv1a_64_lower(String)方法,分別計(jì)算傳入的key(op_id)和定義的java成員變量opId,然后計(jì)算他們是否匹配。

如果匹配的話,就會(huì)覆蓋。

所以,關(guān)鍵就在于TypeUtils.fnv1a_64_lower(String)方法實(shí)現(xiàn),如下:

這個(gè)方法就是如果是'_'或者'-',那么就忽略,也就是如果是op_id,那么就會(huì)變成opid。

如果是大寫,那么就轉(zhuǎn)換成小寫,也就是opId,就會(huì)變成opid。

所以op-id或者op_id,都可以匹配opId或者opid

public static long fnv1a_64_lower(String key){
        long hashCode = 0xcbf29ce484222325L;
        for(int i = 0; i < key.length(); ++i){
            char ch = key.charAt(i);
            if(ch == '_' || ch == '-'){
                continue;
            }
            if(ch >= 'A' && ch <= 'Z'){
                ch = (char) (ch + 32);
            }
            hashCode ^= ch;
            hashCode *= 0x100000001b3L;
        }
        return hashCode;
    }

關(guān)于如何分析fastjson處理下劃線和駝峰問(wèn)題的方法和源碼問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識(shí)。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI