web-dev-qa-db-ja.com

Pythonクライアントを記憶する単純なマルチスレッドソケットサーバーの作成方法

クライアントを記憶し、リクエストごとに新しいソケットを作成しない単純なPythonエコーサーバーを作成するにはどうすればよいですか?同時アクセスをサポートできる必要があります。一度接続できるようにしたいこのクライアントなどを使用して、継続的にデータを送受信します。

import socket

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
Host = raw_input("Server hostname or ip? ")
port = input("Server port? ")
sock.connect((Host,port))
while True:
    data = raw_input("message: ")
    sock.send(data)
    print "response: ", sock.recv(1024)

つまりサーバーをポート50000で実行し、上記のクライアントを使用して、これを実行できるようにします。

me@mine:~$ client.py
Server hostname or ip? localhost
Server Port? 50000
message: testa
response: testa
message: testb
response: testb
message: testc
response: testc
33
nettux

クライアントごとにスレッドを使用してclient.recv()のブロッキングを回避し、新しいクライアントをリッスンするためだけにメインスレッドを使用できます。接続すると、メインスレッドは新しいクライアントをリッスンするだけの新しいスレッドを作成し、60秒間通話しないと終了します。

_import socket
import threading

class ThreadedServer(object):
    def __init__(self, Host, port):
        self.Host = Host
        self.port = port
        self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        self.sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        self.sock.bind((self.Host, self.port))

    def listen(self):
        self.sock.listen(5)
        while True:
            client, address = self.sock.accept()
            client.settimeout(60)
            threading.Thread(target = self.listenToClient,args = (client,address)).start()

    def listenToClient(self, client, address):
        size = 1024
        while True:
            try:
                data = client.recv(size)
                if data:
                    # Set the response to echo back the recieved data 
                    response = data
                    client.send(response)
                else:
                    raise error('Client disconnected')
            except:
                client.close()
                return False

if __name__ == "__main__":
    while True:
        port_num = input("Port? ")
        try:
            port_num = int(port_num)
            break
        except ValueError:
            pass

    ThreadedServer('',port_num).listen()
_

クライアントは、60秒間何も操作しないとタイムアウトし、再接続する必要があります。関数client.settimeout(60)ThreadedServer.listen()行を参照してください

69
nettux