标签导航:

go语言map排序后转json计算md5值与php结果不一致的问题分析

在使用go语言处理map数据时,常常需要进行排序后再将其转换为json格式,并计算其md5值用于数据校验。然而,如果go语言生成的json字符串与php生成的json字符串不一致,则会导致md5值计算结果不同。本文将针对这个问题,分析其原因并提供解决方案。

问题描述:

一段go语言代码尝试对一个map进行字典序排序,然后将其转换为json字符串,最后计算其md5值。然而,其结果与php端计算的md5值不一致。go语言代码如下:

package main

import (
    "fmt"
    "sort"
    "encoding/json"
    "crypto/md5"
    "encoding/hex"
    "bytes"
)

func main() {

    params := make(map[string]string)
    params["name"] = "111"
    params["domain"] = "www.baidu.com?name=1&id=1"

    // 排序
    keys := make([]string, len(params))
    i := 0
    for k, _ := range params {
        keys[i] = k
        i++
    }
    sort.Strings(keys)

    for _, k := range keys {
        fmt.Println(k)
    }

    byteBuf := bytes.NewBuffer([]byte{})
    encoder := json.NewEncoder(byteBuf)
    // 特殊字符不转义
    encoder.SetEscapeHTML(false)
    err := encoder.Encode(params)
    if err != nil {
        panic(err)
    }
    data := byteBuf.String()
    fmt.Println(data)
    h := md5.New()
    // 与 php 一致
    h.Write([]byte(`{"domain":"www.baidu.com?name=1&id=1","name":"111"}`))
    // 不一致
    // h.Write([]byte(data))
    fmt.Println(hex.EncodeToString(h.Sum(nil)))
    // php 结果为 06820973e7b8d3acdb4763264a803620

}

这段代码首先创建了一个map,然后对其键进行排序,之后将排序后的map转换为json字符串,最后计算该字符串的md5值。代码中特意注释掉了一行 h.write([]byte(data)),这是因为直接使用go语言生成的json字符串计算md5值与php端的结果不一致。而使用预设的json字符串 {"domain":"www.baidu.com?name=1&id=1","name":"111"} 计算md5值则与php端一致。

问题根源在于go语言json.encoder的默认行为与php的json编码器在处理特殊字符(例如&)的方式上存在差异。go语言的json.encoder默认会对特殊字符进行转义,而php的json编码器则可能不会。这种差异导致了最终生成的json字符串不同,从而导致md5值计算结果不一致。 通过encoder.setescapehtml(false)虽然避免了html实体的转义,但可能仍存在其他字符转义的差异。 解决方法需要仔细检查go语言和php端json编码器的配置,确保两者对特殊字符的处理方式一致。