Python和Windows命名管道
从Python与Windows上的命名管道通信的正确方式是什么?我已经在谷歌上搜索过了,找不到任何包装此通信的软件包 有:Python和Windows命名管道,python,windows,python-3.x,named-pipes,Python,Windows,Python 3.x,Named Pipes,从Python与Windows上的命名管道通信的正确方式是什么?我已经在谷歌上搜索过了,找不到任何包装此通信的软件包 有: 一些关于如何使用pywin32的描述(不过,我找不到如何使用它连接到现有管道) 这个包裹:(也不走运) 这里有一段代码,我还没有测试过: 我只需要连接到现有的命名管道并对其进行读/写。我以前只尝试过与串行端口通信(使用pySerial),与之相比,在命名管道上能找到的信息是如此之少,我感到惊讶。对于Python来说,通常有很多用于任何目的的指南 我将感谢您的帮助。我成功
- 一些关于如何使用pywin32的描述(不过,我找不到如何使用它连接到现有管道)
- 这个包裹:(也不走运)
- 这里有一段代码,我还没有测试过:
我将感谢您的帮助。我成功地使用了以下片段。此代码源于Wireshark Wiki。它需要包中的
win32pipe
和win32file
我不知道它关闭管道的方式是否100%正确
你提到的-乔纳森·莱因哈特。我尝试过,但无法创建命名管道。我想知道该代码是否只适用于打开已由另一个进程创建的命名管道。为了连接到现有命名管道,您可以使用通过
pywin32
包提供的CreateFile
API。由于我花了一段时间才建立起一个工作基础,这里有一个对我来说很好的示例客户机/服务器(python 3.6.5,Windows 10 Pro x64上的pywin32 223):
示例输出服务器
> python pipe_test.py s
pipe server
waiting for client
got client
writing message 0
writing message 1
writing message 2
writing message 3
writing message 4
writing message 5
writing message 6
writing message 7
writing message 8
writing message 9
finished now
显然,您需要对各种调用进行一些错误检查,但这应该是可行的
附加说明:我的一位同事在客户端尝试对管道执行I/O时遇到了关闭管道的问题(声称“所有管道实例都很忙”的例外)。事实证明,他在客户端代码中使用了
os.path.exists
,以测试命名管道在运行CreateFile
之前是否已经存在。这不知怎么打破了管道。因此,使用上述方法(CreateFile
包装在try-except中)是尝试连接到管道的安全方法,直到服务器端创建管道为止。您应该能够使用f=open(r'\.\pipe\','rb+',buffering=0)
,然后调用f.read(nbytes)
和f.write(string)
@eryksun您的建议与我列表中的第三个解决方案相同,只是它是“r+b”,您需要将字节写入文件,而不是字符串。我自己也希望这是正确的方式,并确实设法通过这种方式与命名管道建立了通信。我以前的错误是,我用错误的方式发送ascii控制代码。“rb+”和“r+b”是相同的。另外,您的问题被标记为Python2,没有提到Python3,所以我自然假设您使用的是Python2,其中str
字符串是字节字符串。在Python3中,您需要编写字节
。明白了。我没有专门为Python2标记它,只是为Python标记。没错,Python标记可以双向使用,但我仍然希望Python3问题被标记为“Python-3.x”。我想这是因为在大多数平台上,python
仍然是python2,而运行python3需要python3
。太棒了!这是我能找到的唯一一个在windows上运行python 3+的管道示例。您不需要对使用win32file.CreateFile()
创建的文件调用一些CloseHandle()
?有没有办法用和实现这一点?@SergeRogatch是的,你可能应该在客户端中加入相同的finally
子句,因为它已经存在于服务器中。在安装pywin32
之后,在Win8.1和Py3.7.6
上运行得很好。虽然有点晚,但是关于:C#和Python。C#代码是服务器,python代码是客户端。在客户端启动之前,服务器必须运行并等待客户端。
import time
import sys
import win32pipe, win32file, pywintypes
def pipe_server():
print("pipe server")
count = 0
pipe = win32pipe.CreateNamedPipe(
r'\\.\pipe\Foo',
win32pipe.PIPE_ACCESS_DUPLEX,
win32pipe.PIPE_TYPE_MESSAGE | win32pipe.PIPE_READMODE_MESSAGE | win32pipe.PIPE_WAIT,
1, 65536, 65536,
0,
None)
try:
print("waiting for client")
win32pipe.ConnectNamedPipe(pipe, None)
print("got client")
while count < 10:
print(f"writing message {count}")
# convert to bytes
some_data = str.encode(f"{count}")
win32file.WriteFile(pipe, some_data)
time.sleep(1)
count += 1
print("finished now")
finally:
win32file.CloseHandle(pipe)
def pipe_client():
print("pipe client")
quit = False
while not quit:
try:
handle = win32file.CreateFile(
r'\\.\pipe\Foo',
win32file.GENERIC_READ | win32file.GENERIC_WRITE,
0,
None,
win32file.OPEN_EXISTING,
0,
None
)
res = win32pipe.SetNamedPipeHandleState(handle, win32pipe.PIPE_READMODE_MESSAGE, None, None)
if res == 0:
print(f"SetNamedPipeHandleState return code: {res}")
while True:
resp = win32file.ReadFile(handle, 64*1024)
print(f"message: {resp}")
except pywintypes.error as e:
if e.args[0] == 2:
print("no pipe, trying again in a sec")
time.sleep(1)
elif e.args[0] == 109:
print("broken pipe, bye bye")
quit = True
if __name__ == '__main__':
if len(sys.argv) < 2:
print("need s or c as argument")
elif sys.argv[1] == "s":
pipe_server()
elif sys.argv[1] == "c":
pipe_client()
else:
print(f"no can do: {sys.argv[1]}")
> python pipe_test.py c
pipe client
no pipe, trying again in a sec
no pipe, trying again in a sec
no pipe, trying again in a sec
message: (0, b'0')
message: (0, b'1')
message: (0, b'2')
message: (0, b'3')
message: (0, b'4')
message: (0, b'5')
message: (0, b'6')
message: (0, b'7')
message: (0, b'8')
message: (0, b'9')
broken pipe, bye bye
> python pipe_test.py s
pipe server
waiting for client
got client
writing message 0
writing message 1
writing message 2
writing message 3
writing message 4
writing message 5
writing message 6
writing message 7
writing message 8
writing message 9
finished now