【问题标题】:Python Sockets - Server only receives once?Python Sockets - 服务器只接收一次?
【发布时间】:2012-12-09 23:56:36
【问题描述】:

我一直在研究 Python 套接字,希望了解如何更好地完成网络编程。现在我正在尝试设置一个连接到服务器的客户端,并且可以在关闭之前向它发送任意数量的消息。我仍然不太了解所有内容,我只能让它发送一条消息。对于有套接字编程经验的人来说,这可能是显而易见的。有人可以向我解释如何让它发送多条消息吗?也许为我提供一个很好的资源来解释连接、绑定、关闭、接收和所有主要的套接字功能是如何工作的? (我发现的大多数资源只是告诉我何时使用它们。)

客户

# !usr/bin/python

import socket
import sys


def main():
    host = ""
    port = 8934
    message = "Hello World!"

    host = raw_input("Enter IP: ")
    #Create Socket
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    except socket.error, msg:
        print "Failed to create socket. Error code: %s Error Message: %s"%(str(msg[0]),msg[1])
        sys.exit()
    print "Socket created"

    #Connect to server
    s.connect((host,port))

    while message != "/e":
        #Send Data
        message = raw_input("Send >> ")
        try:
            s.sendall(message)
        except socket.error, msg:
            print "ERROR %s"%(msg[1])
            print "Failed to send."
        s.close()

if __name__ == "__main__":
    main()

服务器

# !usr/bin/python

import socket
import sys

HOST = ""
PORT = 8934
s = socket.socket()

class BoServer:


    def __init__(self):
        try:
            s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        except socket.error,msg:
            print "Unable to create socket"
            sys.exit()
        print "Socket created."
    def bind(self):
        s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        try:
            s.bind((HOST,PORT))
        except socket.error,msg:
            print "Bind failed. Closing..."
            sys.exit()
        print "Socket bound."
    def run(self):
        while True:
            s.listen(10)
            print "Socket Listening"
            conn, addr = s.accept()
            print "Connected to %s:%s"%(addr[0],addr[1])
            income = conn.recv(4096)
            if income != "":
                print income

def main():
    serv = BoServer()
    serv.bind()
    serv.run()

if __name__ == "__main__":
    main()

【问题讨论】:

  • 你应该只listenaccept一次,然后只在conn.recv上循环。
  • 在您接受()套接字后,您只从新套接字读取一次,然后服务器关闭套接字。
  • 要记住的一件事:即使在您修复此问题之后,您也不能依赖每个recv 获得恰好一个send 的结果。在本地测试时,它似乎在 99.999% 的时间内都可以工作,但是一旦您部署到 Internet,它就会一直失败。您可能会得到半个 send,或 2-1/2 sends。如果您需要单独的消息,则需要使用分隔符(例如,换行符)、长度/类型/任何前缀(例如,网络字符串)或一些自分隔的消息类型(例如,JSON-RPC),并且您需要 @987654330 @ 放入缓冲区并解析它。
  • 另外,如果消息是行,反之亦然,并且缓冲和解析听起来太复杂,请考虑使用makefile,它可以让您调用readlinefor line in f 或其他任何东西,就像一个常规文件。

标签: python sockets networking


【解决方案1】:

在您的客户端,您正在关闭while True: 循环内的套接字,因此您将无法在循环的未来迭代中传输更多消息。如果您打算在不同的连接上发送每条消息,那么您需要在循环中创建套接字。如果您打算在同一连接上发送多条消息,则需要将 close 调用移到循环之外。

在服务器端,您在循环的每次迭代中运行一次listen(),这是不必要的:它的目的是为缓冲的传入连接设置队列长度,因此只需要调用一次。在继续循环的第二次迭代之前,您也只从套接字执行一次读取,在该循环中您接受另一个传入连接。

根据您希望服务器的行为方式,另一个潜在问题是您一次只为一个传入连接提供服务。如果您打算处理长时间运行的连接,那么这可能是一个问题。要一次处理多个连接,您可能希望在其自己的线程上处理每个连接(例如,使用来自SocketServer standard library moduleThreadingTCPServer 类),或者使用像Twisted 这样的异步IO 框架。

【讨论】:

  • 感谢 James,帮助我更好地理解了一点。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-08-10
  • 2016-04-05
  • 1970-01-01
  • 2012-06-20
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多