Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/python/312.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Python 如何:windows上close_fds=True和重定向stdout/stderr的解决方法_Python_Windows_Subprocess - Fatal编程技术网

Python 如何:windows上close_fds=True和重定向stdout/stderr的解决方法

Python 如何:windows上close_fds=True和重定向stdout/stderr的解决方法,python,windows,subprocess,Python,Windows,Subprocess,我遇到了一个问题:使用Python2.7,无法使用 subprocess.Popen([.......], close_fds=True, stdout=subprocess.PIPE, ...) 在windows上,由于限制。在我的例子中,需要使用close\u fds,因为我不希望子流程继承已经打开的文件。这是在库中调用的,这意味着我无法控制已经打开的文件描述符(Nflag) 这是一个固定的 我的问题是:如何使用子流程而不获取 如果重定向stdin/stdout/stderr,则Windo

我遇到了一个问题:使用Python2.7,无法使用

subprocess.Popen([.......], close_fds=True, stdout=subprocess.PIPE, ...)
在windows上,由于限制。在我的例子中,需要使用
close\u fds
,因为我不希望子流程继承已经打开的文件。这是在库中调用的,这意味着我无法控制已经打开的文件描述符(
N
flag)

这是一个固定的

我的问题是:如何使用子流程而不获取

如果重定向stdin/stdout/stderr,则Windows平台上不支持close_fds


下面的答案

默认情况下,Python 3.7+解决了此问题

这显然是一个棘手的问题:答案是在使用
子流程
模块之前,迭代已经打开的文件描述符

def _hack_windows_subprocess():
    """HACK: python 2.7 file descriptors.
    This magic hack fixes https://bugs.python.org/issue19575
    by adding HANDLE_FLAG_INHERIT to all already opened file descriptors.
    """
    # See https://github.com/secdev/scapy/issues/1136
    import stat
    from ctypes import windll, wintypes
    from msvcrt import get_osfhandle

    HANDLE_FLAG_INHERIT = 0x00000001

    for fd in range(100):
        try:
            s = os.fstat(fd)
        except:
            continue
        if stat.S_ISREG(s.st_mode):
            handle = wintypes.HANDLE(get_osfhandle(fd))
            mask   = wintypes.DWORD(HANDLE_FLAG_INHERIT)
            flags  = wintypes.DWORD(0)
            windll.kernel32.SetHandleInformation(handle, mask, flags)
下面是一个没有它会崩溃的示例:

import os, subprocess
f = open("a.txt", "w")
subprocess.Popen(["cmd"], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
f.close()
os.remove(f.name)
回溯(最近一次呼叫最后一次):

模块中第1行的文件“stdin”

WindowsError:[Error 32]进程不符合要求 FICHIER CIER FICHIER EUTILIS(A.TXT)

现在,修复:

import os, subprocess
f = open("a.txt", "w")
_hack_windows_subprocess()
subprocess.Popen(["cmd"], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
f.close()
os.remove(f.name)
工作


希望我在Windows上的Python2.7中提供了帮助,您可以在inheritable上以[N]的形式显式打开文件,例如
f=open(“a.txt”,“wN”)
。在Python3中,这是默认值。这是正确的,上面的示例仅在程序无法控制以前打开过的文件(例如API)时使用,在3.7中也是如此(仍处于beta版本)子流程使用
PROC\u THREAD\u属性\u HANDLE\u列表来约束哪些可继承句柄实际被继承,从而解决了这个问题。