有些事我无法解决问题。我用:
创建了一个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数据报套接字写回发件人?
感谢您的回答
答案 0 :(得分:3)
假设我们有一台服务器:
# 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)
如果客户端连接并发送消息而没有将自己的名称绑定到套接字,如下所示:
# client.py
import socket
SOCKET_FILE = "mysocket-server"
sk = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
sk.sendto("hello", SOCKET_FILE)
sk.close()
然后将匿名发送消息,而不在客户端绑定地址(即addr == None
)。请注意,这与IP数据报套接字不同,IP套数字一旦发送数据就会自动绑定到新地址(即主机地址和端口号)。
对于Unix数据报套接字上的此类匿名消息,客户端没有分配的地址,并且服务器没有机制将返回数据发送给发送方。
最简单的解决方案是客户端将自己的私有名称绑定到套接字:
# 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()
这会自动将客户端绑定到一个新的“抽象套接字地址”,它会模拟已经执行的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
不是一个出色的服务器设计,因为行为不端的客户端可能会阻塞,阻止服务器为任何其他客户端提供服务。真实的服务器可能会使用单独的工作线程来保持在慢客户端运行。
答案 1 :(得分:0)
您需要在两侧绑定(到不同的套接字)。
不会自动为您创建套接字,因此服务器程序没有响应的名称。
>>> from socket import *
>>> sk=socket(AF_UNIX,SOCK_DGRAM,0)
>>> sk.bind('/tmp/abc')
>>> sk.recvfrom(1024)
('hello', '/tmp/abc1')
>>> from socket import *
>>> sk=socket(AF_UNIX,SOCK_DGRAM,0)
>>> sk.bind('/tmp/abc1')
>>> sk.sendto('hello','/tmp/abc')
5