- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
有些事情我无法理解。我创建了一个 unix 数据报套接字:
self.socket = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
self.socket.bind(SOCKET_FILE)
稍后在代码中我接收到写入套接字的消息
data, addr = self.socket.recvfrom(4096)
但是 addr 似乎一直都是 None。但我需要它来发回回复。
我怎样才能使用 unix 数据报套接字实现写回发件人?
谢谢你的回答
最佳答案
假设我们有一个服务器:
# server.py
import os
import socket
SOCKET_FILE = "mysocket-server"
s = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
s.bind(SOCKET_FILE)
data, addr = s.recvfrom(4096)
s.close()
os.unlink(SOCKET_FILE)
print(data, addr)
如果客户端连接并发送消息,但没有将自己的名称绑定(bind)到套接字,如下所示:
# client.py
import socket
SOCKET_FILE = "mysocket-server"
sk = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
sk.sendto("hello", SOCKET_FILE)
sk.close()
然后消息将匿名发送,客户端没有地址绑定(bind)(即 addr == None
)。请注意,这与 IP 数据报套接字不同,IP 数据报套接字会在您发送数据后立即自动绑定(bind)到新地址(即主机地址和端口号)。
对于这种通过 Unix 数据报套接字传输的匿名消息,客户端没有分配地址,服务器也没有机制可以将返回数据发送给发送者。
最简单的解决方案是客户端将自己的私有(private)名称绑定(bind)到套接字:
# client2.py
import os
import socket
SERVER_FILE = "mysocket-server"
CLIENT_FILE = "mysocket-client"
sk = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
sk.bind(CLIENT_FILE)
sk.sendto("hello", SERVER_FILE)
data, addr = sk.recvfrom(4096)
print(data,addr)
sk.close()
os.unlink(CLIENT_FILE)
然后,使用以下修改后的服务器:
# server2.py
import os
import socket
SOCKET_FILE = "mysocket-server"
s = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
s.bind(SOCKET_FILE)
data, addr = s.recvfrom(4096)
if addr is not None:
s.sendto("world", addr)
print(data, addr)
s.close()
os.unlink(SOCKET_FILE)
你可以看到双向通信是可能的。
在 Linux 上,有一个“抽象命名空间”扩展(参见 unix(7)
联机帮助页),这意味着客户端也可以使用 sk.bind("")
,像这样:
# client3.py
import socket
SERVER_FILE = "mysocket-server"
sk = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
sk.bind("")
sk.sendto("hello", SERVER_FILE)
data, addr = sk.recvfrom(4096)
print(data,addr)
sk.close()
这会自动将客户端绑定(bind)到一个新的“抽象套接字地址”,这在某种程度上模拟了 IP 数据报套接字已经完成的工作。
作为替代方法,您可以使用 SOCK_SEQPACKET
代替 SOCK_DGRAM
。这会自动构建双向连接(如 SOCK_STREAM
)但保留消息边界(如 SOCK_DATAGRAM
)。这是一个服务器,它在循环中接受来自客户端的连接,接收并响应来自每个客户端的两个数据包。
# server4.py
import os
import socket
SOCKET_FILE = "mysocket-server"
s = socket.socket(socket.AF_UNIX, socket.SOCK_SEQPACKET)
s.bind(SOCKET_FILE)
s.listen(5)
try:
while True:
(t, _) = s.accept()
print(t.recv(4096))
t.send("sun")
print(t.recv(4096))
t.send("moon")
t.close()
finally:
os.unlink(SOCKET_FILE)
以下客户端演示了响应数据包是分开保存的:
# client4.py
import socket
SERVER_FILE = "mysocket-server"
sk = socket.socket(socket.AF_UNIX, socket.SOCK_SEQPACKET)
sk.connect(SERVER_FILE)
sk.send("hello")
sk.send("goodbye")
print(sk.recv(4096))
print(sk.recv(4096))
sk.close()
在这里,server4.py
不是一个很好的服务器设计,因为行为不当的客户端可能会阻塞,从而阻止服务器为任何其他客户端提供服务。面对缓慢的客户端,真实的服务器可能会使用单独的工作线程来保持运行。
关于python - UNIX 数据报套接字返回数据给发送者,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47001053/
我是iOS的新手,我想更新ViewDidLoad()函数中的文本。 这是我的按钮功能,单击按钮时会发生动画,并将值“1”添加到“resultText.text” - (IBAction)oneB
做了什么 我有一个名为 MyUser 的自定义 User 模型,如 full example for an custom user model 中所述。在文档和一个所谓的 UserProfile 上,
我有一个 NSMenu(应用程序停靠菜单),其中有几个具有相同操作的项目。 如何找出发件人项目(触发操作的项目)在其容器菜单中的索引? (我对标题不感兴趣,因为它可能是重复的) 这就是我尝试过的,但它
我正在开发一个带有 NSTableView 的 macOS 应用程序,我希望能够在用户选择一行时使用 Cmd+C 快捷键复制单元格的内容。我已经实现了该方法 copy(sender: AnyObjec
我一直在使用 MVVM 的 RelayCommand 成功地将操作绑定(bind)到 XAML,但是我的 ItemsControl 有一个小问题。
我的 C# Winform 面板中有一堆文本框。每行文本框的命名如下: tb1 tbNickName1 comboBox1 tb2 tbNickName2 comboBox2 tb3 tbNickNa
我有一个IBAction,例如: - (IBAction)thisThing:(id)sender { [self doSomething]; } 我想这样做(手动调用 IBAction): [s
我知道如何通过 zeromq 将字符串消息从 C++ 发送到 Python。 这是我知道的发送字符串消息的代码: C++ 发件人代码: void *context = zmq_ctx_new(); v
我是一名优秀的程序员,十分优秀!