記一個python字典和json.dumps()的坑


最近項目中需要與管易雲erp做對接,看了他的接口文檔,php的示例代碼,於是用python仿寫。

其中傳的參數data中前面幾個json數據是固定的,最后需要加一個簽名,該簽名是對前面的json數據字符串化后,首尾拼接上screct字符串,再做md5處理(32位大寫),再將該簽名添加到之前的json中作為post參數傳遞過去。

問題就出在組裝json字符串和簽名中,因為python內置的字典是無序的,導致我組裝好的json數據作為參數傳遞給自己編寫的簽名函數時,字典內部的順序是變化的,所以簽名前后的md5值不一樣,導致頻頻報錯,自己起初沒在意數據順序的事,因為對接金蝶erp時沒有簽名這一項,數據傳遞的很簡單,導致排錯時方向出現錯誤。后來想到這了顧開始解決。

解決方法是使用collections庫中的OrderedDict(有序字典)模塊,組裝好的數據就不會亂序,做的md5簽名也就前后一致了

但,在對json數據使用json.dumps()方法時,字符串化后的數據在逗號后會有一個空格,導致md5值出錯

原理:

 

 

 

解決方法是

json.dumps(data,separators=(',',':'))

最后,將自己寫的函數留下來做個例子

def getShops():
    data = OrderedDict()
    data["appkey"] = appkey
    data["sessionkey"] = sessionkey
    data["method"] = method
    data["page_no"] = "1"
    data["page_size"] = "10"
    data["sign"] = sign(data,secret)
    
    response = requests.post(url=url, data=json.dumps(data))
    print(response.text)


def sign(data,secret):
    str = json.dumps(data,separators=(',',':'))
    fullStr = secret+str+secret
    signCode = hashlib.md5(fullStr.encode("utf-8")).hexdigest().upper()
    #print(signCode)
    return signCode

 

 

再訪問新增訂單接口時,又報簽名出錯,按着之前的邏輯去查,發現json.dumps()含中文數據時,會將中文默認轉為ascii碼,從而導致MD5不一致,需轉成中文才行

原理:

def dumps(obj, skipkeys=False, ensure_ascii=True, check_circular=True,
        allow_nan=True, cls=None, indent=None, separators=None,
        default=None, sort_keys=False, **kw):

解決:將上述代碼紅色參數改為False即可。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM