python - 服务器仅与一个客户端(Python Socket)建立连接

标签 python multithreading sockets client-server

首先,我希望我没有写太多。我是新人,我希望没有人有疑问,一切对读者来说都是清楚的。我希望有人能帮助我。

我已经在套接字服务器和客户端工作了几个星期。随着时间的推移,我添加了更多功能。一开始,只是拥有一个回显服务器。之后,我做了一个服务器,它返回时间、随机数或客户端询问的其他特定内容。最后一件事,我添加到脚本中,以便服务器可以接受 2 个客户端,以便它们可以在它们之间进行通信。然而,客户无法写出他想要的消息,因为他总是需要等待第二个客户的答复。当我遇到这个问题时,我了解了线程

因此,我想添加的下一个功能(我被困了大约两三周的地方)是两个客户端可以不间断地互相发送消息,并且不需要像以前那样等待。

我有一个服务器脚本:

import socket
import threading
from datetime import datetime
from random import randint

global num
num = 0

class serverThread(threading.Thread):
    def __init__(self):
        global num
        num = num + 1
        self.id = num
        threading.Thread.__init__(self)
        client, address = server.accept()
        self.client = client
        self.address = address
        print "serverThread init finished-" + str(self.id)

    def run(self):
        print "r1 num-" + str(self.id)
        size = 1024
        while True:
            #try:
                print "r2*************-" + str(self.id)
                data = self.client.recv(size)
                print "r3..... " + data
                print "r4-" + str(self.id)
                if data:
                    print "r5-" + str(self.id)
                    response = data

                    self.client.send(response)
                    print "r6-" + str(self.id)
                else:
                    print "r7-" + str(self.id)
                    raise Exception('Client disconnected-' + str(self.id) )
            #except:
               # print "Except"
               # self.client.close()
               # return

def create(ipHost, port):
    server = socket.socket()
    server.bind((ipHost, port))
    print "The server was created successfully."
    return server

def listen(server):
    server.listen(5)
    c1 = serverThread()
    c1.start()
    c2 = serverThread()
    c2.start()
    print "finished both threads created"

    while c1.isAlive() and c2.isAlive():
        continue

server = create("0.0.0.0", 1729)
listen(server)

如您所见,我没有使用 try except 因为我不知道如何使用它们。

我的第二个脚本,客户端:

import socket
import threading

class sendThread(threading.Thread):
    def __init__(self, ip, port):
        threading.Thread.__init__(self)
        self.client = socket.socket()
        self.port = port
        self.ip = ip
        self.client.connect((self.ip, self.port))
        print "[+] New send thread started for "+ip+":"+str(port) + "...Everything went successful!"

    def run(self):
        while True:
            data = raw_input("Enter command:")
            self.client.send("Client sent: " + data)

class receiveThread(threading.Thread):
    def __init__(self, ip, port):
        threading.Thread.__init__(self)
        self.client = socket.socket()
        self.ip = ip
        self.port = port
        self.client.connect((str(self.ip), self.port))
        print "[+] New receive thread started for "+ip+":"+str(port) + "...Everything went successful!"

    def run(self):
        print "Entered run method"
        size = 1024
        while True:
            data = self.client.recv(size)
            if data != "" or data:
                print "A client sent " + data
def client():
    port = 1729
    ip = '127.0.0.1'
    print "Connection from : "+ip+":"+str(port)

    receive = receiveThread(ip, port)
    print "b1"
    receive.start()
    print "b2"
    send = sendThread(ip, port)
    print "b3"
    send.start()
    while send.isAlive() and receive.isAlive():
        continue
    print "-----END of While TRUE------"
    print "Client disconnected..."

client()

我认为描述我的脚本是一个好主意,在我的代码中逐步进行,也许它会有所帮助,因此它会更具可读性。

服务器脚本

我创建一个套接字服务器,并调用bind 方法。我调用 listen 方法并开始接收客户端。我为每个客户端创建一个线程,我将接受 (accept()) 并从中接收 (recv) 数据。创建每个客户端线程后,我会打印一条消息,表明它们已成功创建。当我启动客户端线程时,它们等待接收发送的消息(recv)并发送它。如果我没记错的话,我只需要 send 方法,而不是告诉发送它。

客户端脚本

客户端将有两个线程。一种用于发送消息(只要您想要),另一种用于始终等待另一个客户端发送的消息。

问题

当我想在运行它打印的两个客户端之前运行服务器时

The server was created successfully.

我运行 2 个客户端并且都打印:

Connection from : 127.0.0.1:1720
[+] New receive thread started for 127.0.0.1:1720...Everything went successful!
b1
Entered run method
b2
[+] New send thread started for 127.0.0.1:1720...Everything went successful!
b3
Enter command:

但是,创建的第二个客户端与服务器之间的连接存在问题。我这样做的是,当服务器收到客户端发送的消息时,它将将该消息打印为服务器中的输出。但是,当第一个客户端发送消息时,它会打印该消息。但当第二个客户端发送时则不然。

我什至尝试复制客户端脚本并将其放入一个新文件中,因此两个客户端来自两个不同的文件,可能会发现问题。然而,这并没有帮助。我尝试运行第一个文件,然后运行第二个文件,反之亦然。总是第二个客户端的服务器出现问题。 (顺便说一句,我还想知道为什么客户端文件不打印他自己发送的消息(他仍然会从服务器接收消息),但这是次要问题)。

我希望我没有做得太长或太远,希望有人可以帮助找到我的代码中的问题。

如果您告诉我代码中存在什么问题,而不是给我您创建或找到的问题,我会更高兴!

最佳答案

我认为这可能是因为您有 2 个线程尝试同时接受连接?

您创建第一个线程,然后该线程的 init 函数通过 socket.accept() 接受连接。然后,在收到连接之前,您立即创建另一个服务器线程,该线程还调用套接字上的accept()。我的猜测是,第二个接受调用未注册,因为 1 个线程已经“锁定”该对象。

与其立即创建 2 个线程,不如尝试仅在有人连接到套接字时才创建一个线程?

client = socket.accept()
serverThread1 = serverThread(client)
serverThread2.start()
client = socket.accept()
serverThread2 = serverThread(client)
serverThread2.start()

serverThread 类现在将客户端套接字作为构造函数参数。

关于python - 服务器仅与一个客户端(Python Socket)建立连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36249170/

相关文章:

python - 如何删除python内置函数?调用某个函数时 min() 消失

c# - 导致竞争条件的多线程

java - .read() 抛出间歇性 SocketTimeoutException,即使数据应该存在

java - 如何查明客户端/服务器套接字连接的所有资源(线程、PrintWriter 等)是否正确关闭?

python - 没有轮询的基于选择的套接字循环

python - Tkinter Widget 颜色不变

python - sqlalchemy在where子句中动态使用and_

java - 在 onDestroy 中执行长时间运行的操作

c# - 在 AutoCAD .Net 插件中执行多线程作业

python pandas plot 系列 matplotlib