1

コンソール チャット プログラムを作成しようとしていますが、ループに問題があります。入力を取得し、同時に他の人の入力を受け取ることはできません。一方の端から 2 つ以上のメッセージが送信された場合、もう一方の端は次のメッセージを送信するまで次のメッセージを受信できません。私はPythonにかなり慣れていないので、正しい方向へのナッジを探していました。私はマルチスレッド化を考えましたが、それは私の理解力から少し外れています. 他のアイデアはありますか?

import EncMod
from socket import *

#Get User Info
Ip = raw_input('IP>>>')
Port = int(raw_input('Port>>>'))
User = raw_input('Username>>>')

#Open Socket To Server
EncCon = socket(AF_INET, SOCK_STREAM)
EncCon.connect((Ip, Port))

print '\nStarting Chat....'
print '\n<-------------------------------------------->\n\n'

#Send/Receive Loop
while 1:
   MsgOut = raw_input()
   if MsgOut: EncCon.send(MsgOut)

   MsgIn = EncCon.recv(1024)
   if MsgIn: print MsgIn

EncCon.close()
4

3 に答える 3

0

Twistedフレームワークは、このタスクの実行を支援するために使用できます。以下のコードは、サーバーインスタンスの設定に基づいて、クライアントが接続したり通信したりできるチャットサーバーを開始します。要件に合わせて適切な変更を加えることができます。

from twisted.internet.protocol import Factory
from twisted.protocols.basic import LineReceiver
from twisted.internet import reactor

class Chat(LineReceiver):

    def __init__(self, users):
        self.users = users
        self.name = None
        self.state = "GETNAME"

    def connectionMade(self):
        self.sendLine("What's your name?")

    def connectionLost(self, reason):
        if self.users.has_key(self.name):
            del self.users[self.name]

    def lineReceived(self, line):
        if self.state == "GETNAME":
            self.handle_GETNAME(line)
        else:
            self.handle_CHAT(line)

    def handle_GETNAME(self, name):
        if self.users.has_key(name):
            self.sendLine("Name taken, please choose another.")
            return
        self.sendLine("Welcome, %s!" % (name,))
        self.name = name
        self.users[name] = self
        self.state = "CHAT"

    def handle_CHAT(self, message):
        message = "<%s> %s" % (self.name, message)
        for name, protocol in self.users.iteritems():
            if ':' in message:
                self.exc(message.split(':')[0])
            if protocol != self:
                protocol.sendLine(message)

    def exc(self, cmd):
        print cmd
        if cmd == 'who':
            for i in self.users:
                print i


class ChatFactory(Factory):

    def __init__(self):
        self.users = {} # maps user names to Chat instances

    def buildProtocol(self, addr):
        return Chat(self.users)


reactor.listenTCP(8123, ChatFactory())
reactor.run()
于 2012-07-05T22:39:52.780 に答える
0

問題は、データが受信されるまで recv() 呼び出しがブロックされ、recv() がブロックされている間、プログラムは stdin からの入力があるかどうかを確認していないことです。これに対する従来のシングルスレッドの解決策は、(EncCon.setblocking(False) を介して) ソケットを非ブロック I/O に設定し、代わりに select() 内にプログラム ブロックを配置することです。EncCon と stdin の両方を select() に ( read-socket-set 引数の一部として) 渡して、 select() のいずれかがあなたに与えるデータを持っているときはいつでも戻るようにします。(Windows では select() が stdin でブロックすることを許可していないため、このメソッドは Windows では機能しないことに注意してください :P )

于 2012-07-06T02:22:49.500 に答える
0

スレッド化は思ったほど難しくなく、それをマスターすることはツールチェストに非常に貴重な追加となります。

Thread のサブクラスであるクラスを作成し、run() メソッドがあることを確認してください。次に、クラスをインスタンス化し、その start() メソッドを呼び出します。

糸を止めることは、正しく行うのがより困難です。フラグを設定し、while ループで定期的にチェックすることを確認するのが最善です。そのため、recv() のブロッキングには、たとえば 1 秒のタイムアウトが必要です。

from socket import *
from threading import Thread


#Get User Info
Ip = raw_input('IP>>>')
Port = int(raw_input('Port>>>'))
User = raw_input('Username>>>')

#Open Socket To Server
EncCon = socket(AF_INET, SOCK_STREAM)
EncCon.connect((Ip, Port))

print '\nStarting Chat....'
print '\n<-------------------------------------------->\n\n'


class ReceiveThread(Thread):

    def __init__(self, sock):
        Thread.__init__(self)
        self.sock = sock
        self.shouldstop = False

    def run(self):
        self.sock.settimeout(1)
        while not self.shouldstop:
            try:
                data = self.sock.read()
                print data
            except socket.timeout:
                continue

    def stop(self):
        self.shouldstop = True


# start receive loop:
r = ReceiveThread(EncCon).start()


#Send Loop
while 1:
    MsgOut = raw_input()
    if MsgOut: EncCon.send(MsgOut)

    if MsgOut == '.':
        r.stop()
        r.join()


EncCon.close()

現在、このプログラムには、リッスンせずにすぐに接続するため、2 つのインスタンスを開始できないという元の問題が残っています。しかし、それはあなたの質問の主要な部分ではなかったと思います。

于 2012-07-06T02:35:21.420 に答える