- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我以前设法实现了一个客户端服务器套接字脚本,该脚本在单个客户端和服务器之间中继消息,现在我正在尝试实现一个多客户端系统。
更具体地说,我想将服务器用作两个客户端之间的某种媒介,它从一个客户端检索信息并将其中继到另一个客户端。我曾尝试附加并发送接收客户端的端口号,然后从服务器端的消息中提取它。之后,我会尝试将其发送到具有该端口号的任何套接字,但遇到了一些麻烦(因为端口号是在我确定的发送点确定的?),所以现在我只是在尝试将发送的消息中继回去给所有客户。但是,问题在于该消息仅被发送到服务器,而没有被中继到所需的客户端。
我以前曾尝试实现对等系统,但遇到麻烦,因此我决定退后一步,改为这样做。
Server.py:
import socket, _thread, threading
import tkinter as tk
SERVERPORT = 8600
HOST = 'localhost'
class Server():
def __init__(self):
self.Connected = True
self.ServerSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.ServerSocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR,1)
self.ServerSocket.bind((HOST, SERVERPORT))
self.ServerSocket.listen(2)
self.Clients = []
def Listen(self):
print('Server is now running')
while self.Connected:
ClientSocket, Address = self.ServerSocket.accept()
self.Clients.append(Address)
print('\nNew user connected', Address)
t = threading.Thread(target=self.NewClient, args=(ClientSocket,
Address))
t.daemon = True
t.start()
self.Socket.close()
def NewClient(self, ClientSocket, Address):
while self.Connected:
if ClientSocket:
try:
ReceivedMsg = ClientSocket.recv(4096)
print('Message received from', Address, ':', ReceivedMsg)
self.Acknowledge(ClientSocket, Address)
if ReceivedMsg.decode('utf8').split()[-1] != 'message':
ReceiverPort = self.GetSendPort(ReceivedMsg)
self.SendToClient(ClientSocket,ReceivedMsg,ReceiverPort)
except:
print('Connection closed')
raise Exception
ClientSocket.close()
def Acknowledge(self, Socket, Address):
Socket.sendto(b'The server received your message', Address)
def GetSendPort(self, Msg):
MsgDigest = Msg.decode('utf8').split()
return int(MsgDigest[-1])
def SendToClient(self, Socket, Msg, Port):
Addr = (HOST, Msg)
for Client in self.Clients:
Socket.sendto(Msg, Client)
def NewThread(Func, *args):
if len(args) == 1:
t = threading.Thread(target=Func, args=(args,))
elif len(args) > 1:
t = threading.Thread(target=Func, args=args)
else:
t = threading.Thread(target=Func)
t.daemon = True
t.start()
t.join()
Host = Server()
NewThread(Host.Listen)
import socket, threading
import tkinter as tk
Username = 'Ernest'
PORT = 8601
OtherPORT = 8602
SERVERPORT = 8600
HOST = '127.0.0.1'
class Client():
def __init__(self, Username):
self.Connected, self.Username = False, Username
self.Socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
def Connect(self):
print('Trying to connect')
try:
self.Socket.connect((HOST, SERVERPORT))
self.Connected = True
print(self.Username, 'connected to server')
Msg = MsgUI(self.Username)
Msg.Display()
except Exception:
print('Could not connect to server')
raise Exception
def SendMsg(self):
if self.Connected:
Msg = '{} sent you a message {}'.format(self.Username, OtherPORT)
self.Socket.sendall(bytes(Msg, encoding='utf8'))
self.GetResponse()
def GetResponse(self, *args):
AckMsg = '\n{} received the message'.format(self.Username)
NMsg = '\n{} did not receive the message'.format(self.Username)
if self.Connected:
Msg = self.Socket.recv(4096)
print(Msg)
if Msg:
self.Socket.sendall(bytes(AckMsg, encoding='utf8'))
else:
self.Socket.sendall(bytes(NMsg, encoding='utf8'))
class MsgUI():
def __init__(self, Username):
self.Username = Username
self.entry = tk.Entry(win)
self.sendbtn = tk.Button(win, text='send', command=Peer.SendMsg)
def Display(self):
self.entry.grid()
self.sendbtn.grid()
win.mainloop()
win = tk.Tk()
Peer = Client(Username)
Peer.Connect()
GetResponse
方法,而不是
if self.Connected
,我使用
while self.Connected
,但它仍然无法正常工作。
import socket, _thread, threading
import tkinter as tk
SERVERPORT = 8600
HOST = 'localhost'
class Server():
def __init__(self):
self.Connected = True
self.ServerSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.ServerSocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR,1)
self.ServerSocket.bind((HOST, SERVERPORT))
self.ServerSocket.listen(2)
self.Clients = {}
def ConnectClient(self, Username, Port):
Socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.Clients[Username] = [Socket, Port, False]
try:
self.Clients[Username][0].connect((HOST, SERVERPORT))
self.Clients[Username][2] = True
print('Opened port for user', Username)
return Socket
except Exception:
print('Could not open port for user', Username)
raise Exception
def Listen(self):
print('Server is now running')
while self.Connected:
ClientSocket, Address = self.ServerSocket.accept()
print('\nNew user connected', Address)
t = threading.Thread(target=self.NewClient, args=(ClientSocket,
Address))
t.daemon = True
t.start()
self.Socket.close()
def NewClient(self, ClientSocket, Address):
while self.Connected:
if ClientSocket:
try:
ReceivedMsg = ClientSocket.recv(4096)
if b'attempting to connect to the server' in ReceivedMsg:
ClientSocket.send(b'You are now connected to the server')
else:
print('Message received from', Address, ':',ReceivedMsg)
#self.Acknowledge(ClientSocket, Address)
ReceiverPort = self.GetSendPort(ReceivedMsg)
if ReceiverPort != None:
self.SendToClient(ClientSocket,ReceivedMsg,
ReceiverPort)
except:
print('Connection closed')
raise Exception
ClientSocket.close()
def Acknowledge(self, Socket, Address):
Socket.sendto(b'The server received your message', Address)
def GetSendPort(self, Msg):
MsgDigest = Msg.decode('utf8').split()
try:
Port = int(MsgDigest[-1])
except ValueError:
Port = None
return Port
def SendToClient(self, Socket, Msg, Port):
Addr = (HOST, Port)
Receiver = None
for Client, Vars in self.Clients.items():
if Vars[1] == Port:
Receiver = Client
self.Clients[Receiver][0].sendto(Msg, Addr)
def NewThread(Func, *args):
if len(args) == 1:
t = threading.Thread(target=Func, args=(args,))
elif len(args) > 1:
t = threading.Thread(target=Func, args=args)
else:
t = threading.Thread(target=Func)
t.daemon = True
t.start()
t.join()
Host = Server()
if __name__ == '__main__':
NewThread(Host.Listen)
import socket, threading, Server
import tkinter as tk
Username = 'Ernest'
PORT = 8601
OtherPORT = 8602
SERVERPORT = 8600
HOST = '127.0.0.1'
class Client():
def __init__(self, Username):
self.Connected, self.Username = False, Username
def Connect(self):
print('Requesting to connect to server')
try:
self.Socket = Server.Host.ConnectClient(self.Username, PORT)
self.Connected = Server.Host.Clients[self.Username][2]
Msg = '{} is attempting to connect to the server'.format(self.Username)
self.Socket.sendall(bytes(Msg, encoding='utf8'))
ReceivedMsg = self.Socket.recv(4096)
print(ReceivedMsg)
Msg = MsgUI(self.Username)
Msg.Display()
except Exception:
print('Could not connect to server')
raise Exception
def SendMsg(self):
try:
if self.Connected:
Msg = '{} sent you a message {}'.format(self.Username,OtherPORT)
self.Socket.sendall(bytes(Msg, encoding='utf8'))
self.GetResponse()
except Exception:
print('Connection closed')
raise Exception
def GetResponse(self, *args):
AckMsg = '\n{} received the message'.format(self.Username)
NMsg = '\n{} did not receive the message'.format(self.Username)
if self.Connected:
Msg = self.Socket.recv(4096)
print(Msg)
if Msg:
self.Socket.sendall(bytes(AckMsg, encoding='utf8'))
else:
self.Socket.sendall(bytes(NMsg, encoding='utf8'))
class MsgUI():
def __init__(self, Username):
self.Username = Username
self.entry = tk.Entry(win)
self.sendbtn = tk.Button(win, text='send', command=Peer.SendMsg)
def Display(self):
self.entry.grid()
self.sendbtn.grid()
win.mainloop()
win = tk.Tk()
Peer = Client(Username)
Peer.Connect()
KeyError
字典仍然为空,因此我收到了
Clients
。当在客户端文件中对服务器进行函数调用时,很明显,对字典的更新发生在客户端文件中,而不是服务器文件中-这是在另一个实例中。我需要一种更改
Clients
词典内容的方法,该方法由客户端文件调用以起作用,但在服务器文件中生效。
最佳答案
您致力于多线程吗?线程不能在python中同时运行(由于GIL),虽然它们是处理并发操作的一种方法,但它们不是唯一的方法,通常不是最佳的方法,除非它们是唯一的方法。考虑一下这段代码,它不能很好地处理失败案例,但似乎可以作为一个起点。
import socket, select, Queue
svrsock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
svrsock.setblocking(0)
svrsock.bind(('', 17654))
svrsock.listen(16)
client_queues = {}
write_ready=[] # we'll update this for clients only that have things in the queue
while client_queues.keys() + [svrsock] :
readable, writable, exceptional = select.select(client_queues.keys() + [svrsock] , write_ready, [])
for rd in readable:
if rd is svrsock: # reading listening socket == accepting connection
conn, addr = svrsock.accept()
print("Connection from {}".format(addr))
conn.setblocking(0)
client_queues[conn] = Queue.Queue()
else:
data = rd.recv(1024)
if data:
# TODO: send to all queues
print("Message from {}".format(rd.getpeername()))
for sock, q in client_queues.iteritems():
q.put("From {}: {}".format( rd.getpeername(), data))
if sock not in write_ready:
write_ready.append(sock)
for rw in writable:
try:
data = client_queues[rw].get_nowait()
rw.send(data)
except Queue.Empty:
write_ready.remove(rw)
continue
write_ready
列表中,其中有待处理的数据。然后,每个准备好写入的套接字都将其下一个排队的消息写入其中。如果没有更多消息,则将收件人从
write_ready
列表中删除。
select
。原则上讲,这很容易。我们向
select()
传递了一个可能的套接字读取列表,另一个可能的套接字写入列表以及最终列表,对于这个简化的演示,我完全忽略了它们。 select调用的结果将包括一个或多个实际上已准备好进行读取或写入的套接字,这使我可以阻塞直到一个或多个套接字准备好进行 Activity 为止。然后,我每遍都处理所有准备好进行 Activity 的套接字(但是它们已经被筛选成不会阻塞的套接字)。
$ python ./svr.py
Connection from ('127.0.0.1', 52059)
Connection from ('127.0.0.1', 52061)
Message from ('127.0.0.1', 52061)
Message from ('127.0.0.1', 52059)
Message from ('127.0.0.1', 52059)
$ nc localhost 17654
hello
From ('127.0.0.1', 52061): hello
From ('127.0.0.1', 52059): hello
From ('127.0.0.1', 52059): hello
$ nc localhost 17654
From ('127.0.0.1', 52061): hello
hello
From ('127.0.0.1', 52059): hello
hello
From ('127.0.0.1', 52059): hello
select
为什么比并发执行更具吸引力,请考虑以下因素:Apache基于线程模型,换句话说,每个连接都有一个工作线程。 nginx基于
select
模型,因此您可以看到它可能有多快。更不用说nginx本质上更好,因为Apache受益于线程模型,因为它大量使用模块来扩展功能(例如mod_php),而nginx没有此限制,并且可以处理来自任何线程的所有请求。但是通常认为nginx的原始性能要高得多,效率也要高得多,主要原因是它避免了apache固有的几乎所有cpu上下文切换。这是一种有效的方法!
select
可以很好地扩展(cpu的速度比网络快得多,因此可能不是瓶颈)。
关于python - 具有中央中继服务器的多线程套接字,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53937597/
我正在尝试为登录实现突变。突变验证提供的 id_token 并通过 session 登录用户。突变本身有效(使用 GraphiQL 验证),但我在将其与 Relay 集成时遇到了问题。 当用户登录时,
这个问题已经有答案了: Return a JSON string explicitly from Asp.net WEBAPI? (9 个回答) 已关闭 5 年前。 我在 VS 中创建了一个非常简单的
这个问题已经有答案了: Return a JSON string explicitly from Asp.net WEBAPI? (9 个回答) 已关闭 5 年前。 我在 VS 中创建了一个非常简单的
是否可以通过 .net 应用程序从使用 smtp-relay 的服务器发送电子邮件。 我正在使用 app.config 获取服务器 IP 的实际值,以及电子邮件应使用的地址。 根据 IT 技术人员的说
我有一个服务器和一个安卓应用程序。我的服务器想要请求一个目的地,但要使用运行我的应用程序的 android 设备的 IP。因此,我需要在我的应用程序中代理请求(中继它们),以便目的地可以看到客户端的
我正在尝试构建一个接受多个客户端连接的简单 TCP 中继。 Client1 --> TCPrelay1 --> RealServer1 --> TCPrelay1 --> Client1 Client
我有以下情况: SomeServer(S) (C)MyApp(S) (C)User (S) represents a server socket (C) represents a client s
我尝试 relay-modern 有一段时间了,我想知道 createFragmentContainer 除了描述 的片段之外还有什么用途>应该属于组件。 例如这就是文档显示它应该如何显示的方式 Pa
我正在尝试在本地设置一个简单的中继/映射服务器,感觉必须有一些现成的解决方案,但我似乎找不到。 我正在调试我的一个需要连接到 host_A 的应用程序。我不想连接到 host_A,而是想将其配置为连接
我有一个使用 CodeIgniter 构建并由共享托管服务提供商托管的应用程序。 我的 MX 记录指向 Gmail 服务器,我的应用程序可以使用 Gmail 和标准 CodeIgniter 电子邮件方
我正在使用 Azure 中继、混合连接处理程序测试 Azure 事件网格。这对我不起作用。 我可以看到消息正在发布到 Azure 事件网格。到目前为止,一切都很好。 我已设置对事件网格主题的订阅,并将
已关闭。这个问题是 off-topic 。目前不接受答案。 想要改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 已关闭10 年前。 Improve th
当我尝试 pod repo update 时出现此错误和 pod install CDN: trunk URL couldn't be downloaded:https://cdn.cocoapods
我正在开发一个 WebApp,想知道在方法之间共享 JDBC 连接对象是否有任何优点/缺点 所以情况是 这是一个 WebApp 和多用户环境(大约 1000 个用户同时使用) 我正在使用支持连接池的容
我正在尝试使用 graphene-django 的 DjangoFilterConnectionField 实现以下目标: { allUsers(username_Icontains: "abc"
我正在尝试在 css 中为以下内容设置动画。我读过以前的帖子,但 settimeout 似乎不起作用,或者它似乎没有给我预期的结果。 我正在尝试制作一个简单的动画,它每次将行高更改 0.5,我试过循环
我有 KMtronics USB 继电器,我想用我的 Android 设备控制它。是否可以运行以下 Linux 命令: #!/bin/sh # sends on signal to the USB r
我们的应用程序全部采用 HTTPS,但我们从不支持 SSL 的低预算设备获取 mjpeg 流。 因此,这意味着我们可以:我。混合内容并让浏览器崩溃。 (这是 Not Acceptable ,因为我们训
操作系统:CentOS Python:2.4 和 2.7 我正在尝试为我的应用程序构建 TCP 和 UDP 流量的中继或代理。 我的设置是 app1 通过不断发送 UDP 数据包来验证 app2 是否
例如我有连接类型: let usersType = new GraphQLObjectType({ name: 'Users', description: 'users array',
我是一名优秀的程序员,十分优秀!