1

我创建了一个 Python 项目来测试 google map api。我在测试第一个 API 时遇到了问题:附近的搜索请求。

我试过这个官方文档提供的网址: https ://maps.googleapis.com/maps/api/place/nearbysearch/json?location=-33.8670522,151.1957362&radius=500&types=food&name=harbour&sensor=false&key=AddYourOwnKeyHere 如果我把这个网址在我的浏览器上,我会在浏览器上获得完美的数据。然后我什么也没做,只创建了一个 python 模块,代码如下:

if __name__ == '__main__':  

import socket
import ssl 


sock = ssl.wrap_socket(socket.socket()) 
sock.connect(('maps.googleapis.com', 443)) 

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect(('maps.googleapis.com', 443))
sslSocket = socket.ssl(s)
uri = 'GET /maps/api/place/nearbysearch/json?location=-33.8670522,151.1957362&radius=500&sensor=false&key=MyKey HTTP/1.1\r\nHost:maps.googleapis.com\r\nContent-Type: application/json; charset=UTF-8\r\naccept-encoding:gzip,deflate,sdch\r\n\r\n'
sslSocket.write(uri)

data = sock.read()
print 'going to connect...'
print data 
s.close()

不幸的是,我在控制台上得到的结果是这样的:</p>

going to connect...
HTTP/1.1 200 OK
Content-Type: application/json; charset=UTF-8
Date: Wed, 02 Oct 2013 08:05:59 GMT
Expires: Wed, 02 Oct 2013 08:10:59 GMT
Cache-Control: public, max-age=300
Vary: Accept-Language
Server: mafe
X-XSS-Protection: 1; mode=block
X-Frame-Options: SAMEORIGIN
Alternate-Protocol: 443:quic
Transfer-Encoding: chunked

560b
{
"debug_info" : [],
"html_attributions" : [
"Listings by \u003ca href=\"http://www.yellowpages.com.au/\"\u003eYellow           Pages\u003c/a\u003e"],
"next_page_token" : "ClRHAAAA4lOaMlZcUjyndIqhBsHzyyU0nD5PHarmjgmnguj3morn7Em-    ipqhSR33V__d0kjrL4PzAqY7y4TAK7Uj3XzWjhoUchLBcBs-    TNFxWIvGst8SEArmj_T7_eDZWyr8UDGy5OEaFPFaK2yHDg7Kv2U3Obsy7opkQfEw",
"results" : [
{
"geometry" : {
"location" : {
"lat" : -33.8599827,
"lng" : 151.2021282
},
"viewport" : {
"northeast" : {
"lat" : -33.8552584,
"lng" : 151.2031401
},

这个结果是不完整的,我试了很多次,结果都是一样的。那么,如何解决这个问题呢?

4

4 回答 4

1

问题是您正在为 HTTP 使用套接字。具体来说,sock.read()不保证能读取对方发送的所有字节;相反,您必须不断询问它,直到它返回一个空字节字符串。然后你可以确定你已经收到了所有东西。但是,套接字真的很棘手,使用它们几乎从来都不是一个好主意,而选择更高级别的框架几乎总是一个好主意。例如,evenssl_sock.write()可能实际上甚至不会写入所有内容(或者至少sock.send()肯定会以这种方式运行)而是返回发送的字节数,如果这小于您传递给它的数据,则您必须重试其余的 (未发送)数据。

现在,使用 HTTP,我以前从未见过有人使用套接字。而是使用urllib2甚至更好的Python Requestsurllib2.urlopen(...)之类的东西。此外,还有urllibhttplib。例如,使用Python Requests,您的代码变得如此简单:

import requests

response = requests.get('https://maps.googleapis.com/maps/api/place/nearbysearch/json?location=-33.8670522,151.1957362&radius=500&sensor=false&key=MyKey')
data = response.json()  # shortcut to `import json; json.loads(response.text)`

安装 Python 请求的最简单方法是使用pip install requests(或easy_install requests)。或者,如果您更愿意避免依赖:

import json
import urllib

data_raw = urllib.urlopen('https://maps.googleapis.com/maps/api/place/nearbysearch/json?location=-33.8670522,151.1957362&radius=500&sensor=false&key=MyKey').read()
data = json.loads(data_raw)

现在,data保证包含整个响应;你永远不会得到部分回应。

PS如果您需要在请求中传递一些 HTTP 请求标头,您也可以这样做:

requests.get(url, headers={...})

PPS如果您想了解更多关于原始套接字的信息(希望仅用于非 HTTP 目的),请参阅The Python Socket Programming HOWTO

于 2013-10-02T10:22:39.500 回答
0

help(sock.read)给出:

关于模块 ssl 中读取方法的帮助:

ssl.SSLSocket 实例的 read(self, len=1024) 方法 最多读取 LEN 个字节并返回它们。在 EOF 上返回零长度字符串。

并且您的回复接近 1024 个字符,因此请尝试:

data = '  '
while len(data) > 0:
    data = sock.read()
    print data,
print "\nDone."
于 2013-10-02T10:32:58.253 回答
0

数据在多次发送和接收中发送。所以你应该运行一个循环来检查数据何时到来

receivedString = ""
While True:
    data = socket.recv(1024) #max length 1024
    #time.sleep(0.25)
    if(len(data) < 1):
       break
    recievedString = recievedString + data
于 2016-01-17T12:00:10.153 回答
-1

第一个答案很酷。但有时你必须使用原始套接字来解决问题......

所以我使用下面的代码尝试了我的 API Key

data = '  '
while len(data) > 0:
    data = sock.read()
    print data,
    print len(data)
print "\nDone."

虽然有时数据的长度是 1024,有时是 381,我不知道为什么。另外,程序不会跳出循环,这也让我很困惑。

PS谁能解释一下响应内容开头的“560b”是什么意思,非常感谢。

于 2013-10-03T11:08:13.560 回答