本文阐述Go语言如何复现PHP和Python中字典排序及签名生成的功能。 PHP和Python代码分别使用http_build_query和urllib.parse.urlencode进行URL编码,并计算MD5签名。 关键在于字典排序和Unicode字符的处理。
PHP代码示例:
$params = [ 'y' => 1, 'x' => '有中文有符!@ 2号空格http://aaa.com', 'a' => '{"a1":222,"a2":"u4e2du6587"}', ]; $signkey = 'abc'; ksort($params); $encodedparams = http_build_query($params); echo md5($encodedparams . $signkey);
Python代码示例:
import hashlib, urllib.parse def sort_by_key(dict_var): sorted_tuple = sorted(dict_var.items(), key = lambda x:x[0]) sorted_dict = {} for tuple_item in sorted_tuple: sorted_dict[tuple_item[0]] = tuple_item[1] return sorted_dict sign_key = 'abc' params = { 'y': 1, 'x': '有中文有符!@ 2号空格http://aaa.com', 'a': '{"a1":222,"a2":"u4e2du6587"}', } params = sort_by_key(params) md5 = hashlib.md5() sign_str = '%s%s' % (urllib.parse.urlencode(params), sign_key) md5.update(sign_str.encode('utf-8')) sign = md5.hexdigest() print(sign)
Go语言实现中,利用sort包排序字典键,net/url包进行URL编码,crypto/md5包计算MD5哈希值。 需要注意的是,PHP中的uxxxx表示普通字符串,而Go和Python会将其识别为Unicode转义字符,需要正确处理。
Go语言代码实现:
package main import ( "crypto/md5" "encoding/hex" "fmt" "net/url" "sort" ) func main() { params := map[string]string{ "x": "有中文有符!@ 2号空格http://aaa.com", "y": "1", "a": "{"a1":222,"a2":"\u4e2d\u6587"}", } keys := make([]string, 0, len(params)) for k := range params { keys = append(keys, k) } sort.Strings(keys) data := url.Values{} for _, k := range keys { data.Add(k, params[k]) } encodedParams := data.Encode() signKey := "abc" signData := encodedParams + signKey hash := md5.New() hash.Write([]byte(signData)) bytes := hash.Sum(nil) sign := hex.EncodeToString(bytes) fmt.Println(sign) }
此Go代码与PHP代码功能一致,正确处理了Unicode字符,生成的MD5签名与PHP代码输出相同。 Python代码结果与PHP不同,源于Unicode字符处理方式的差异。 Go代码通过对\u4e2d\u6587的正确处理,确保了与PHP代码一致的签名结果。