前提
現在サーバー側Windows搭載のPC
クライアント側ラズパイ4が3台
このような構成になっています
実現したいこと
実現したことしては
- サーバで値を入力する。
2)3台のクライアントに同じ値を同時に送信 - 3台のクライアントで処理を実施。
- すべての結果を収集。
- 1に戻る
最終的には、PC(サーバー)からの指令が来るたびにラズパイ(クライアント)が同時にプログラムを実行し
実行結果をサーバに返したいです
発生している問題・エラーメッセージ
エラーメッセージ
現在のプログラムだと一度通信を切らないと、サーバー側の数字が更新できない
whileループの中に入れると、複数台同時に送信できない、あるいは
Bad adressといったエラーがでる
該当のソースコード
サーバー側
python
1 2import socket 3import threading 4import time 5 6IPADDR = "" 7PORT = 8sock_sv = socket.socket(socket.AF_INET) 9sock_sv.bind((IPADDR, PORT)) 10sock_sv.listen(3) 11 12def recv_client(sock, addr): 13 try: 14 sock.send(signal.encode("utf-8")) 15 data = sock.recv(4096) 16 time.sleep(2) 17 print(data) 18 19 except ConnectionResetError: 20 pass 21 22while True: 23 signal=input("signal=") 24 25 sock_cl1, addr1 = sock_sv.accept() 26 sock_cl2, addr2 = sock_sv.accept() 27 sock_cl3, addr3 = sock_sv.accept() 28 29 thread1 = threading.Thread(target=recv_client, args=(sock_cl1, addr1)) 30 thread2 = threading.Thread(target=recv_client, args=(sock_cl2, addr2)) 31 thread3 = threading.Thread(target=recv_client, args=(sock_cl3, addr3)) 32 # スレッド処理開始 33 thread1.start() 34 thread2.start() 35 thread3.start() 36 #スレッド終了待ち合わせ 37 thread1.join() 38 thread2.join() 39 thread3.join() 40
クライアント側
python
1import socket 2import queue 3import threading 4import numpy as np 5import time 6 7IPADDR = "" 8PORT = 9sock =socket.socket(socket.AF_INET) 10sock.connect((IPADDR,PORT)) 11signal=0 12q= queue.Queue() 13 14def recv_data(sock): 15 while True: 16 try: 17 signal_data = sock.recv(1024) 18 print(signal_data.decode("utf-8")) 19 q.put(signal_data) 20 except ConnectionResetError: 21 break 22 sock.shutdown(socket.SHUT_RDWR) 23 sock.close() 24 25thread = threading.Thread(target=recv_data, args=(sock,)) 26thread.start() 27 28while True: 29 signal=q.get() 30 signal=float(signal) 31 try: 32 if signal > 0: 33 ser = serial.Serial("/dev/ttyS0", 38400,bytesize=7) 34 msg =ser.readline() 35 msg = msg.decode() 36 tok = msg.split(',') 37 S_data=tok[1]+","+tok[2]+","+tok[4] 38 S_data=str(S_data) 39 S_data=S_data+"signal= "+str(signal) 40 sock.send(S_data.encode("utf-8")) 41 42 else: 43 data=("error") 44 sock.send(data.encode("utf-8")) 45 except ConnectionResetError: 46 break 47 48sock.shutdown(socket.SHUT_RDWR) 49sock.close()
試したこと
ループ外にinputを設けるとエラーは起きないが、数字が更新できない
thread.start()の上にinputを設けると個々に数字を送るので同時に実行できない
補足情報(FW/ツールのバージョンなど)
ここにより詳細な情報を記載してください。
実現したい処理の流れがよくわかりません。 特にサーバ(PC)側での値の入力。
1) サーバで値を入力する。
2) 3台のクライアントで処理を実施。
3) すべての結果を収集。
4) 1に戻る
でしょうか? それとも入力はクライアントごとに行ないたいのでしょうか?
処理の流れはTakaiYさんのおっしゃる通りです
入力は同じものを全クライアントを同時に送信したいです
socket.ioのタグは外した方がいいです。あと初質問じゃないですよね?
前の同じような質問が削除申請中みたいです。
クライアント3台から接続した後、例えばサーバー側から開始コマンドを決めて送ればいいかと思います。
クライアント側は送られたコマンドが開始コマンドなら処理を開始する、とかでしょうか・・・
サーバー側はacceptするスレッドを作成し、接続したソケットをリストか何かに並べておき、メインスレッドで開始の入力待ちをします。入力されたらスレッドを起動して接続したソケットの数分起動して、それぞれのスレッドで開始コマンドを送る、といった感じです。クライアントは今のままでいいでしょう。
すみません,前回の質問は編集方法が分からなかったので,新しく上げなおしたので全く同じものです
実現したいのはおっしゃる通りの手法です.クライアントの数だけスレッドを用意するイメージですか?
あとメインループではなく,新しくスレッドで分けるってことですか?
> クライアントの数だけスレッドを用意するイメージですか?
私が言ったのは(メインスレッドの他に)「クライアントの数+1」ですね
> 新しくスレッドで分けるってことですか?
多分Yesです
inputより前にacceptすることで、TCPの3Way Handshakeを事前に済ませています。1パケット届けばクライアントは処理を開始できます。
スレッドでもいいですが、
https://docs.python.org/ja/3/library/selectors.html
https://docs.python.org/ja/3/library/select.html
などを使うことで、1スレッドでソケットや標準入力を一度に扱う手もあります。
>>dameo
acceptを別スレッドを移行せずに無事に同時に処理を開始することができましたが,
「inputより前にacceptすることで、TCPの3Way Handshakeを事前に済ませています。1パケット届けばクライアントは処理を開始できます。」
これをする利点がつかみ切れていません,知識不足で申し訳ないのですが,ご教授してもらえると助かります.
クライアント側からの送信を接続を切らずに止めたいのですが,上記のことで解決できますでしょうか?
利点は「同時に送信」の精度を上げる目的です。必要な精度が分からなかったので。
「クライアント側からの送信を接続を切らずに止めたい」はちょっと意味が分かりません。
現在稼働しているときは,
1)サーバで値を入力する。
2)3台のクライアントに同じ値を同時に送信
3) 3台のクライアントで処理を実施。
4) すべての結果を収集。
5) 1に戻る
1に戻れているのですが,クライアントから何度も結果が送られているので
サーバ側から信号があれば,クライアントから一度(あるいは一定時間)のみ返事が来るようにしたいです
プログラムを再起動とかは避けたいです
すみません。プロトコルを決めれば済む話かと思いますが、それは別の質問ですね。
お手数ですが、別質問としてあげて他の方に聞いてください。
「クライアントから何度も結果が送られている」は、何が送られてきているんでしょうか?
singal=1として入力するとsignal=1を利用した処理結果が送られ続けます
signalは新しく入力を要求していますが,その間も結果が流れているので,信号入力毎に一度の処理といった形を実現したいです.
>>68user
ありがとうございます.スレッド処理よりも良い点が分かれば使ってみます
二度目ですが、当初の質問の回答は得られているので、ベストアンサーを選び、別の質問にしてください。
分かりました,ありがとうございます
回答1件
あなたの回答
tips
プレビュー