【问题标题】:'socket.error: [Errno 9] Bad file descriptor' as I run my client'socket.error: [Errno 9] Bad file descriptor' 在我运行客户端时
【发布时间】:2015-12-12 10:20:54
【问题描述】:

当我运行我的客户端时,我的服务器崩溃了,

上面写着:

Traceback (most recent call last):
      File "C:/Users/Or/PycharmProjects/2.6/cyber.py", line 17, in <module>
        data = client_socket.recv(1024)
      File "C:\Python27\lib\socket.py", line 174, in _dummy
        raise error(EBADF, 'Bad file descriptor')
    socket.error: [Errno 9] Bad file descriptor

服务器:

import socket
import random
import time

server_socket = socket.socket()
server_socket.bind(('0.0.0.0', 8820))
server_socket.listen(1)

(client_socket, client_address) = server_socket.accept()

TIME = time.asctime(time.localtime())
NAME = "Frank"
RAND = random.randrange(1,10)
EXIT = client_socket.close()
options = [TIME, NAME, RAND, EXIT]

data = client_socket.recv(1024)
if data == str(TIME):
    client_socket.send('Time is: ' + TIME)
elif data == str(NAME):
    client_socket.send('My name is: ' + NAME)
elif data == str(RAND):
    client_socket.send(RAND)
elif data == str(EXIT):
    EXIT




client_socket.close()
server_socket.close()

客户:

import socket

my_socket = socket.socket()
my_socket.connect(('127.0.0.1', 8820))

theData = raw_input("enter command")
options = ['TIME', 'NAME', 'RAND', 'EXIT']
for z in options:
    if theData == z:
        my_socket.send(theData)
        data = my_socket.recv(1024)
        print data
    else:
        theData = raw_input("enter a real command")

我该怎么办?

【问题讨论】:

  • 迭代一个封闭的端口。哪个结束套接字会话,服务器,客户端?永远服务器为什么?如何检查套接字错误或错误?不要在服务器上使用自动关闭(单个命令或消息)功能。定义关闭所有操作的关闭命令(例如:套接字已使用)。

标签: python python-2.7 sockets errno


【解决方案1】:

您的服务器中的以下行 blocks 直到您的客户端连接:

(client_socket, client_address) = server_socket.accept()

在您的客户端连接后,您的服务器会执行以下操作:

  .... = client_socket.close()

然后你的服务器会这样做:

 client_socket.recv(1024)

那是行不通的。

我该怎么办?

删除这一行:

EXIT = client_socket.close()

您的代码还有其他几个问题:

1) 你不知道send() 将实际发送多少数据:

socket.send(bytes[, flags])
...
应用程序负责检查所有数据是否已发送; 如果只传输了部分数据,应用程序需要 尝试传递剩余的数据。 https://docs.python.org/3/library/socket.html#socket.socket.send

换句话说,send() 不一定能够一次性发送整个消息——因此 send() 返回实际发送的字节数。您可以使用sendall() 来改善这个问题。

2) 但是,对于send()sendall(),您不知道您的数据是以 10 个块、5 个块还是全部以 1 个块的形式发送的,因此您必须使用循环来发送 @987654331 @整条消息:

data = []

while True:
    chunk = my_socket.recv()
    if not chunk: break  #When the other side closes the socket, chunk = ''
    data.append(chunk)

all_data = ''.join(data)

当对方关闭socket时,recv()会返回一个空字符串。

3) 现在是大的:

data = client_socket.recv(1024)

您的服务器必须在循环中使用recv() 以确保它从客户端读取整个消息:

data = []

while True:
    chunk = client_socket.recv()
    if not chunk: break  #When the other side closes the socket, chunk = ''
    data.append(chunk)

all_data = ''.join(data)

但是读取循环如何知道它何时读取了所有数据?请注意,如所写,循环在另一端关闭套接字时终止,导致recv() 返回一个空白字符串,从而终止while 循环。但是,在您的情况下,客户端在其send() 之后不会关闭套接字,因此服务器的recv() 将永远不会返回空白字符串,并且while 循环将继续尝试recv()--但客户端将拥有发送完消息,等待回复。死锁。

【讨论】:

    猜你喜欢
    • 2023-01-24
    • 2021-12-05
    • 2011-12-02
    • 1970-01-01
    • 2016-04-15
    • 2019-08-19
    • 1970-01-01
    • 2020-02-25
    • 2021-12-20
    相关资源
    最近更新 更多