当前位置: 首页 > news >正文

requests 发送一个 json 格式的 post 请求

今天给一位同学解决post发送数据格式为json格式的请求,顺便确认一下问题归属。

背景:

用postman工具发送一个数据格式为json的请求,得到了服务器的响应。

用python的requests库写的请求,却报错了。没有得到该有的结果。

解决方法:
  1. 先确认自己的请求信息和函数使用正确。包括请求头、请求体和参数传递。

  1. 看服务端的日志,如果服务端的日志当中收到了这2个请求的请求数据,看日志中提示什么错误并比对一下2份请求数据的不同,顺便有需要找开开聊聊。

基于此,来说说requests中发送json格式的处理。

1、json数据格式的请求头

在post请求当中,json数据格式的请求,请求头常规为:Content-Type:application/json

2、requests库中 post请求的json参数

  post请求方法:post(url,data=None,json=None,**kwargs)

python的requests的post请求中,有一个json参数。源码中对于此参数的说明如下:

一个json序列化的python对象。python中字典表达与json很相似。

在post请求中,对传进来的json值,会做如下处理:

  1、会使用json模块中的dumps方法转成json数据。

  2、会增加消息头中的content_type为application/json

所以,json参数不需要提前使用json模块的方法转成json字符串。

请注意,这里有坑:如果在传参时,提前转换成json字符串:requests.request("post",url,json=json.dumps(a),headers=headers)

在post请求中,还会再使用jsons模块转成json数据。那么此时的请求数据会在最外多了一层引号。

请求数据结果为:"{\"pwd\": \"1234567890\", \"mobilephone\": \"18611000001\"}"

而实际上我们要发送的数据是没有外层引号的,即:{\"pwd\": \"1234567890\", \"mobilephone\": \"18611000001\"}

源码如下:

3、示例代码(仅为用法示例。演示接口并不支持application/json格式):
importrequestsa={"mobilephone":"18611000001","pwd":"xxxxxxxxxxxx"}url="http://XXXXXXXX"#消息头指定headers={'Content-Type':'application/json;charset=UTF-8'}#发送post请求 json参数直接为一个字典数据。res=requests.request("post",url,json=a,headers=headers)print(res.status_code)print(res.text)

打印发送出去的请求数据,请求的结果:


sessions.py499行打印发送请求数据:请求头为:{'User-Agent':'python-requests/2.19.1','Accept-Encoding':'gzip, deflate','Content-Type':'application/json;charset=UTF-8','Content-Length':'51','Connection':'keep-alive','Accept':'*/*'}请求体为:b'{"pwd": "1234567890", "mobilephone": "18611000001"}'200{"status":0,"code":"20103","data":null,"msg":"手机号不能为空"}

json 请求中固定键名顺序 & 消除键和值之间的空格

实际工作中遇到了以下2种情况。

  1. 服务端要求json字符串,键名的顺序固定

  1. 服务端对于接收到的json数据中,若key和value之间有空格,则解析不了。

第1种情况:服务端要求json字符串,键名的顺序固定

服务端在解析客户端请求时,要求收到的请求json数据中,键名的顺序要固定 。比如第一个键名必须是mac,第二个键名必须是agentCode等。

而我们使用requests发送请求数据时,我们是对字典进行json处理的,顺序并不是固定的。

处理方法:在给requests传参时,就给固定顺序的字典就好。

使用collections.OrderedDict。它是有序字典,记住了键值对的添加顺序。

请注意:如果初始化的时候同时传入多个参数,它们的顺序是随机的,不会按照位置顺序存储。

示例代码:


importcollectionscontent=collections.OrderedDict()content["mac"]="NDU1N2RkOTRiYjQ3ZDI5YzI0ZmI5YTQ3ZjMxZGU0OTc2YWY2ZTc3Zg=="content["agentcode"]="100001"content["msgbody"]={"customercode":"02000003","sourceinfo":[{"SourceCode":"10001","startdate":"20190601","enddate":"20190601"}]}print(content)#输出结果:按键名添加的顺序输出OrderedDict([('mac','NDU1N2RkOTRiYjQ3ZDI5YzI0ZmI5YTQ3ZjMxZGU0OTc2YWY2ZTc3Zg=='),('agentcode','100001'),('msgbody',{'sourceinfo':[{'SourceCode':'10001','startdate':'20190601','enddate':'20190601'}],'customercode':'02000003'})])

通过使用orderedDict处理之后,将content作为requests请求中json参数值。那么发给服务器端的数据,就是固定的键名顺序。

处理之后,服务器收到的请求数据:

第2种情况:服务端对于接收到的json数据中,若key和value之间有空格,则解析不了。

在解决了固定顺序键名问题之后,可能你还会遇到,后台开发大佬 跟你说:不行啊,你这键名和键值之间有空格,我们不支持解析。

这种情况下,要么你让开发改代码,要么你自己发送的请求中去掉空格。如果你说服不了开发改,那就只能自己处理啦。自己的处理的话,请继续往下看。

此乃空格:

首先,找原因。空格是怎么来的??

在requests库的源码当中,发送出去的请求数据,默认键名和键值之间都是带空格的。

在源码当中,对传进来的json参数,使用json库的dumps函数转换成json对象,而dumps函数默认设置了键名和键值之间的留有一个空格。

所以,要消除键与值之间的空格,需要在requests的源码当中,将参数转换成json对象时,设置separators的值去掉空格。这样发往服务器的数据中键名和值之间就没有空格了。

修改源码如下:

在requests源码的models.py文件中,找到prepare_body函数,修改如下图片中,红色框框中的内容:指定json中键名和键值之间无空格

修改完成之后,再次向服务器发送json数据格式的post请求,服务器收到的数据如下(可以看到键名和值之间没有空格了哦。。):

json 请求中中文乱码处理

最近收到一个问题:json格式请求数据中有中文,导致服务端签名失败。

问题详情:

一位同学在发送json格式的post请求时,请求数据中有中文内容:

{"inputCodes":["6932608700850"],"terminal":{"status":1,"channel":"D002","storeCode":"2107","passage":"D002","storeName":"重百超市黄泥塝店","identity":"","maxProductCount":5,"posId":"D002"}}

header={"client_id":"DataSync","sign":"46BA170CFC30C571358E59EDDA63B506","Content-Type":"application/json;charset=UTF-8"}

在使用requests库的post请求发送出去之后,服务端收到的不是中文,导致签名失败,数据如下:

{"terminal": {"status": 1, "channel": "D002", "identity": "", "passage": "D002", "maxProductCount": 5, "posId": "D002", "storeName": "\u91cd\u767e\u8d85\u5e02\u9ec4\u6ce5\u585d\u5e97", "storeCode": "2107"}, "inputCodes": ["6932608700850"]}

希望在服务端中收到的数据中,中文仍然是中文,因为有些服务端并没有此做处理。

解决方法:

requests库中,在处理json格式的请求时调用的json.dumps方法参数ensure_ascii默认为True.表示序列化时对中文默认使用的ascii编码。

如果想要显示中文,则将此参数的值改为False即可。

源码修改:

在requests源码的models.py文件中,找到prepare_body函数。找到如下图中的代码,在comlexjson.dumps(json)里加个参数ensure_ascii=False.

保存源码的修改之后,再次运行,在服务器端就能看到中文啦。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Node.js实现一个文章生成器
  • YOLOv8改进 | 注意力机制 | 结合静态和动态上下文信息的注意力机制
  • 2024年6月份找工作和面试总结
  • RabbitMQ 更改服务端口号
  • 力扣1895.最大的幻方
  • 51单片机嵌入式开发:3、STC89C52操作8八段式数码管原理
  • NativeMemoryTracking查看java内存信息
  • udp发送数据如果超过1个mtu时,抓包所遇到的问题记录说明
  • 9 redis,memcached,nginx网络组件
  • 单/多线程--协程--异步爬虫
  • 洛谷 P2141 [NOIP2014 普及组] 珠心算测验
  • Harris点云关键点检测
  • 三、docker配置阿里云镜像仓库并配置docker代理
  • 使用瀚高数据库开发管理工具进行数据的备份与恢复---国产瀚高数据库工作笔记008
  • 使用Python绘制堆积柱形图
  • [nginx文档翻译系列] 控制nginx
  • es的写入过程
  • mysql_config not found
  • PermissionScope Swift4 兼容问题
  • react-core-image-upload 一款轻量级图片上传裁剪插件
  • SpringCloud集成分布式事务LCN (一)
  • vuex 笔记整理
  • WePY 在小程序性能调优上做出的探究
  • 阿里云前端周刊 - 第 26 期
  • 飞驰在Mesos的涡轮引擎上
  • 欢迎参加第二届中国游戏开发者大会
  • 基于Mobx的多页面小程序的全局共享状态管理实践
  • 浅析微信支付:申请退款、退款回调接口、查询退款
  • 首页查询功能的一次实现过程
  • 微服务框架lagom
  • 为视图添加丝滑的水波纹
  • 一道面试题引发的“血案”
  • 用 vue 组件自定义 v-model, 实现一个 Tab 组件。
  • 在Mac OS X上安装 Ruby运行环境
  • 看到一个关于网页设计的文章分享过来!大家看看!
  • d²y/dx²; 偏导数问题 请问f1 f2是什么意思
  • [Shell 脚本] 备份网站文件至OSS服务(纯shell脚本无sdk) ...
  • C# - 为值类型重定义相等性
  • #android不同版本废弃api,新api。
  • #define与typedef区别
  • (1)安装hadoop之虚拟机准备(配置IP与主机名)
  • (13)DroneCAN 适配器节点(一)
  • (k8s中)docker netty OOM问题记录
  • (附源码)ssm教材管理系统 毕业设计 011229
  • (回溯) LeetCode 77. 组合
  • (六)软件测试分工
  • (篇九)MySQL常用内置函数
  • (转)jdk与jre的区别
  • (转)ORM
  • (转)我也是一只IT小小鸟
  • (轉)JSON.stringify 语法实例讲解
  • **CI中自动类加载的用法总结
  • .NET LINQ 通常分 Syntax Query 和Syntax Method
  • .net操作Excel出错解决
  • .net程序集学习心得