如何以JSON格式发送POST请求?


105
data = {
        'ids': [12, 3, 4, 5, 6 , ...]
    }
    urllib2.urlopen("http://abc.com/api/posts/create",urllib.urlencode(data))

我想发送POST请求,但是其中一个字段应该是数字列表。我怎样才能做到这一点 ?(JSON?)


1
不过,这是否不是数字列表?
Waynn Lue 2012年

如果不知道API需要什么样的输入,就无法回答。
Niklas B.

1
API服务器将@WaynnLue作为字符串而不是列表来获取。
TIMEX 2012年

1
我是否必须将标头设置为“ application / json”或其他内容?
TIMEX 2012年

Answers:


154

如果您的服务器期望POST请求为json,那么您将需要添加标头,并为请求序列化数据...

Python 2.x

import json
import urllib2

data = {
        'ids': [12, 3, 4, 5, 6]
}

req = urllib2.Request('http://example.com/api/posts/create')
req.add_header('Content-Type', 'application/json')

response = urllib2.urlopen(req, json.dumps(data))

Python 3.x

https://stackoverflow.com/a/26876308/496445


如果不指定标题,它将是默认application/x-www-form-urlencoded类型。


我有个问题。是否可以在标头中添加多个项目...如内容类型和客户端ID ... @jdi
Omar

@OmarJandali,add_header()对于要添加的每个标头,只需再次调用即可。
jdi

我有以下代码,但未打印任何内容。它应该打印url和标头,但什么也没打印... req = urllib.Request('http://uat-api.synapsefi.com') req.add_header('X-SP-GATEWAY', 'client_id_asdfeavea561va9685e1gre5ara|client_secret_4651av5sa1edgvawegv1a6we1v5a6s51gv') req.add_header('X-SP-USER-IP', '127.0.0.1') req.add_header('X-SP-USER', '| ge85a41v8e16v1a618gea164g65') req.add_header('Content-Type', 'application/json') print(req)...
Omar Jandali

urllib2无法识别,所以我只使用了urllib。我也收到请求错误。The view tab.views.profileSetup didn't return an HttpResponse object. It returned None instead. @jdi
Omar

@OmarJandali,请记住,此答案最初于2012年在python 2.x下给出。您正在使用Python3,因此导入将有所不同。现在应该是import urllib.requesturllib.request.Request()。此外,打印req对象没有任何意义。您可以清楚地看到通过打印添加了标题req.headers。除此之外,我也不奇怪为什么它在您的应用程序中不起作用。
jdi 2017年


66

对于python 3.4.2,我发现以下将起作用:

import urllib.request
import json      

body = {'ids': [12, 14, 50]}  

myurl = "http://www.testmycode.com"
req = urllib.request.Request(myurl)
req.add_header('Content-Type', 'application/json; charset=utf-8')
jsondata = json.dumps(body)
jsondataasbytes = jsondata.encode('utf-8')   # needs to be bytes
req.add_header('Content-Length', len(jsondataasbytes))
print (jsondataasbytes)
response = urllib.request.urlopen(req, jsondataasbytes)

1
Python3.6.2可行。仅添加req.add_header(...)的标头对我有用。
沙林LK

18

这非常适合 Python 3.5,如果URL中包含查询字符串/参数值,

请求网址= https://bah2.com/ws/rest/v1/concept/
参数值= 21f6bb43-98a1-419d-8f0c-8133669e40ca

import requests

url = 'https://bahbah2.com/ws/rest/v1/concept/21f6bb43-98a1-419d-8f0c-8133669e40ca'
data = {"name": "Value"}
r = requests.post(url, auth=('username', 'password'), verify=False, json=data)
print(r.status_code)

7
在你的代码剪断,标题变量住宿闲置
shookees

4

您必须添加标题,否则会出现http 400错误。该代码在python2.6,centos5.4上运行良好

码:

    import urllib2,json

    url = 'http://www.google.com/someservice'
    postdata = {'key':'value'}

    req = urllib2.Request(url)
    req.add_header('Content-Type','application/json')
    data = json.dumps(postdata)

    response = urllib2.urlopen(req,data)

2

这是一个如何使用Python标准库中的urllib.request对象的示例。

import urllib.request
import json
from pprint import pprint

url = "https://app.close.com/hackwithus/3d63efa04a08a9e0/"

values = {
    "first_name": "Vlad",
    "last_name": "Bezden",
    "urls": [
        "https://twitter.com/VladBezden",
        "https://github.com/vlad-bezden",
    ],
}


headers = {
    "Content-Type": "application/json",
    "Accept": "application/json",
}

data = json.dumps(values).encode("utf-8")
pprint(data)

try:
    req = urllib.request.Request(url, data, headers)
    with urllib.request.urlopen(req) as f:
        res = f.read()
    pprint(res.decode())
except Exception as e:
    pprint(e)

1

在最新的请求包中,您可以使用jsonin requests.post()方法来发送json dict,并且Content-Typein标头将设置为application/json。无需显式指定标头。

import requests

payload = {'key': 'value'}

requests.post(url, json=payload)

请注意,这将导致带单引号的POSTed json,这在技术上是无效的。
杰思罗

@Jethro使用单引号时是否观察到错误?在Python中使用单引号是有效的。就个人而言,我还没有遇到任何与此相关的问题。
jdhao

抱歉,我想错了,我以为我的服务器正在接收单引号JSON,但事实证明这是一个单独的问题,并且存在一些误导性的调试。干杯,这比必须手动指定标头要整整齐齐!
Jethro

0

这对我使用api来说效果很好

import requests

data={'Id':id ,'name': name}
r = requests.post( url = 'https://apiurllink', data = data)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.