1

在 Linux 中。我有一个 ac 程序,它读取一个 2048Byte 的文本文件作为输入。我想从 Python 脚本启动 c 程序。我希望 Python 脚本将文本字符串作为参数交给 c 程序,而不是将文本字符串写入文件以供 c 程序读取。

Python 程序如何启动 ac 程序并为其提供约 2K(文本)数据结构?

另请注意,我不能使用“subprocess.check_output()”。我必须使用“os.system()”。那是因为后者允许我的 c 程序直接访问终端输入/输出。前者没有。

4

1 回答 1

5

您可以将其作为参数传递,只需...将其作为参数传递。大概你想引用它而不是将它作为需要转义的任意数量的参数传递等等,但这很容易使用shlex.quote. 例如:

with open('bigfile.txt', 'rb') as infile:
    biginput = infile.read(2048)
os.system('cprogram {}'.format(shlex.quote(biginput)))

如果你得到一个关于参数的错误或者命令行对于 shell 来说太长了……那么你就不能这样做。Python 不能让 shell 做它不能做的事情,并且你拒绝绕过 shell(我认为是因为误解,但我们暂时忽略它)。因此,您将需要其他方式来传递数据。

但这并不意味着您必须将其存储在文件中。subprocess您可以像 from 一样轻松地使用 shell os.system,这意味着您可以将其传递给子进程的标准输入:

with subprocess.Popen('cprogram {}'.format(shlex.quote(biginput)),
                      shell=True, stdin=subprocess.PIPE) as p:
    p.communicate(biginput)

由于您使用的是shell=True,而不是替换stdoutor stderr,它将获得与使用 完全相同的终端os.system。因此,例如,如果它正在执行,例如,isatty(fileno(stdout))如果您的 Python 脚本在 tty 中运行,则为 true,否则为 false。


附带说明一下,将其存储在 a 中的tempfile.NamedTemporaryFile成本可能不会像您预期的那么高。特别是,子进程可能能够直接从内存磁盘缓存中读取您写入的数据,而不是等待它被刷新到磁盘(并且它可能永远不会被刷新到磁盘)。


我怀疑您认为无法使用的原因subprocess是您在check_output需要时使用了check_call.

如果您使用check_output(或者如果您显式传递stdout=PIPE给大多数其他subprocess函数),则子进程的标准输出是您正在读取的管道,因此它显然不是 tty。

这是有道理的:要么你想捕获输出,在这种情况下C程序不能输出到tty,或者你想让C程序输出到tty,在这种情况下你不能捕获它。*所以,不要捕获输出,一切都会好起来的。


如果我是对的,这意味着你一开始就没有理由使用 shell,这让一切变得容易多了。当然,即使没有 shell,您的数据仍可能大于最大系统参数大小**或资源限制***。在大多数现代系统上,您至少可以指望 64KB,所以一定要先尝试一下:

subprocess.check_call(['cprogram', biginput])

但是如果你得到一个E2BIG错误:

with subprocess.Popen(['cprogram', biginput], stdin=subprocess.PIPE) as p:
    p.communicate(biginput)

* 当然,除非您想为您的子进程伪造一个 tty,在这种情况下,您需要查看os.forkpty相关函数或pty模块。

** 在大多数 *BSD 和相关系统上,sysctl kern.argmax和/或getconf ARG_MAX会给你系统限制,或者sysconf(_SC_ARG_MAX)来自 C。也可能有一个常量ARG_MAX可以通过<limits.h>. 在 linux 上,事情有点复杂,因为有许多不同的限制(其中大多数非常非常高),而不仅仅是一个限制。查看您平台的手册页以execve了解详细信息。

*** 在某些平台上,包括最近的 linux,RLIMIT_STACK会影响您可以传递的最大 arg 大小。再次,请参阅您平台的execve联机帮助页。

于 2013-08-01T20:00:48.360 回答