【问题标题】:Creating a simple Chat application in Python (Sockets)用 Python(套接字)创建一个简单的聊天应用程序
【发布时间】:2016-07-03 18:56:12
【问题描述】:

我正在尝试使用套接字 (python) 创建一个简单的聊天应用程序。客户端可以向服务器发送消息,服务器只需将消息广播给所有其他客户端,发送消息的客户端除外。

客户端有两个线程,永远运行

send: Send 只是将客户端消息发送到服务器。

receive:从服务器接收消息。

服务器也有两个线程,永远运行

accept_cleint:接受来自客户端的传入连接。

broadcast_usr:接受来自客户端的消息并将其广播给所有其他客户端。

但是我得到了错误的输出(请参考下图)。所有线程都假设一直处于活动状态,但有时客户端可以发送消息,有时却不能。例如,Tracey 发送了 4 次“嗨”但未广播,当约翰说“再见”2 次然后 1 次其消息被广播。服务器上似乎有一些thread synchronization 问题,我不确定。请告诉我有什么问题。

下面是代码。

chat_client.py

import socket, threading

def send():
    while True:
        msg = raw_input('\nMe > ')
        cli_sock.send(msg)

def receive():
    while True:
        sen_name = cli_sock.recv(1024)
        data = cli_sock.recv(1024)

        print('\n' + str(sen_name) + ' > ' + str(data))

if __name__ == "__main__":   
    # socket
    cli_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # connect
    HOST = 'localhost'
    PORT = 5023
    cli_sock.connect((HOST, PORT))     
    print('Connected to remote host...')
    uname = raw_input('Enter your name to enter the chat > ')
    cli_sock.send(uname)

    thread_send = threading.Thread(target = send)
    thread_send.start()

    thread_receive = threading.Thread(target = receive)
    thread_receive.start()

chat_server.py

import socket, threading

def accept_client():
    while True:
        #accept    
        cli_sock, cli_add = ser_sock.accept()
        uname = cli_sock.recv(1024)
        CONNECTION_LIST.append((uname, cli_sock))
        print('%s is now connected' %uname)

def broadcast_usr():
    while True:
        for i in range(len(CONNECTION_LIST)):
            try:
                data = CONNECTION_LIST[i][1].recv(1024)
                if data:
                    b_usr(CONNECTION_LIST[i][1], CONNECTION_LIST[i][0], data)
            except Exception as x:
                print(x.message)
                break

def b_usr(cs_sock, sen_name, msg):
    for i in range(len(CONNECTION_LIST)):
        if (CONNECTION_LIST[i][1] != cs_sock):
            CONNECTION_LIST[i][1].send(sen_name)
            CONNECTION_LIST[i][1].send(msg)

if __name__ == "__main__":    
    CONNECTION_LIST = []

    # socket
    ser_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # bind
    HOST = 'localhost'
    PORT = 5023
    ser_sock.bind((HOST, PORT))

    # listen    
    ser_sock.listen(1)
    print('Chat server started on port : ' + str(PORT))

    thread_ac = threading.Thread(target = accept_client)
    thread_ac.start()

    thread_bs = threading.Thread(target = broadcast_usr)
    thread_bs.start()

【问题讨论】:

  • 我认为问题出在您的客户端线程循环中,尽管您的服务器也需要能够处理客户端断开连接。
  • 仅供参考,Twisted 是一个用于制作线程服务器的 Python 库。

标签: python multithreading sockets synchronization chat


【解决方案1】:

好的,我之前在评论中撒了谎,抱歉。问题实际上出在服务器上的broadcast_usr() 函数中。它在recv() 方法中阻塞,并阻止除了当前选定的用户之外的所有用户在通过for 循环时一次说话。为了解决这个问题,我更改了 server.py 程序,为它接受的每个客户端连接生成一个新的 broadcast_usr 线程。我希望这会有所帮助。

import socket, threading

def accept_client():
    while True:
        #accept    
        cli_sock, cli_add = ser_sock.accept()
        uname = cli_sock.recv(1024)
        CONNECTION_LIST.append((uname, cli_sock))
        print('%s is now connected' %uname)
        thread_client = threading.Thread(target = broadcast_usr, args=[uname, cli_sock])
        thread_client.start()

def broadcast_usr(uname, cli_sock):
    while True:
        try:
            data = cli_sock.recv(1024)
            if data:
                print "{0} spoke".format(uname)
                b_usr(cli_sock, uname, data)
        except Exception as x:
            print(x.message)
            break

def b_usr(cs_sock, sen_name, msg):
    for client in CONNECTION_LIST:
        if client[1] != cs_sock:
            client[1].send(sen_name)
            client[1].send(msg)

if __name__ == "__main__":    
    CONNECTION_LIST = []

    # socket
    ser_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # bind
    HOST = 'localhost'
    PORT = 5023
    ser_sock.bind((HOST, PORT))

    # listen    
    ser_sock.listen(1)
    print('Chat server started on port : ' + str(PORT))

    thread_ac = threading.Thread(target = accept_client)
    thread_ac.start()

    #thread_bs = threading.Thread(target = broadcast_usr)
    #thread_bs.start()

【讨论】:

  • 您的代码运行良好,但有一个小错误。发送者必须发送 2 条消息才能广播。似乎客户端的sen_name = cli_sock.recv(1024)data = cli_sock.recv(1024) 都在接收发件人姓名和消息。请参考以下屏幕截图postimg.org/image/3mnwmuyvj
【解决方案2】:

我试图解决你说的@Atinesh 的错误。客户端将被询问一次用户名,此“uname”将包含在要发送的数据中。看看我对 'send' 函数做了什么。

为了便于可视化,我在所有收到的消息中添加了一个“\t”。

import socket, threading

def send(uname):
    while True:
        msg = raw_input('\nMe > ')
        data = uname + '>' + msg
        cli_sock.send(data)

def receive():
    while True:
        data = cli_sock.recv(1024)
        print('\t'+ str(data))

if __name__ == "__main__":   
    # socket
    cli_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # connect
    HOST = 'localhost'
    PORT = 5023

    uname = raw_input('Enter your name to enter the chat > ')

    cli_sock.connect((HOST, PORT))     
    print('Connected to remote host...')


    thread_send = threading.Thread(target = send,args=[uname])
    thread_send.start()

    thread_receive = threading.Thread(target = receive)
    thread_receive.start()

您还必须相应地修改您的服务器代码。

server.py

import socket, threading

def accept_client():
    while True:
        #accept    
        cli_sock, cli_add = ser_sock.accept()
        CONNECTION_LIST.append(cli_sock)
        thread_client = threading.Thread(target = broadcast_usr, args=[cli_sock])
        thread_client.start()

def broadcast_usr(cli_sock):
    while True:
        try:
            data = cli_sock.recv(1024)
            if data:
               b_usr(cli_sock, data)
         except Exception as x:
            print(x.message)
            break

def b_usr(cs_sock, msg):
    for client in CONNECTION_LIST:
        if client != cs_sock:
            client.send(msg)

if __name__ == "__main__":    
    CONNECTION_LIST = []

    # socket
    ser_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # bind
    HOST = 'localhost'
    PORT = 5023
    ser_sock.bind((HOST, PORT))

    # listen    
    ser_sock.listen(1)
    print('Chat server started on port : ' + str(PORT))

    thread_ac = threading.Thread(target = accept_client)
    thread_ac.start()

服务器端发生的变化是:连接的用户和说话的用户不再可见。如果您的目的是连接客户,我不知道这是否意味着那么多。也许如果你想通过服务器严格监控客户端,可能还有另一种方式。

【讨论】:

  • 服务器代码第17行有缩进错误。只需将其向左移动 1 个字符即可。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-10-08
  • 2019-05-19
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多