- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我正在寻找一种从 python 脚本运行外部进程并在执行期间打印其标准输出消息的方法。
下面的代码有效,但在运行时不打印标准输出。当它退出时,我收到以下错误:
sys.stdout.write(nextline) TypeError:must be str,not bytes
p = subprocess.Popen(["demo.exe"],stdout = subprocess.PIPE, stderr= subprocess.PIPE)
# Poll process for new output until finished
while True:
nextline = p.stdout.readline()
if nextline == '' and p.poll() != None:
break
sys.stdout.write(nextline)
sys.stdout.flush()
output = p.communicate()[0]
exitCode = p.returncode
我正在使用 python 3.3.2
最佳答案
Python 3 处理字符串有点不同。最初只有一种类型字符串:str
。当 unicode 在 90 年代获得关注时,新的 unicode
类型已添加以处理 Unicode,而不会破坏预先存在的代码1。这是实际上与 str
相同,但支持多字节。
在 Python 3 中有两种不同的类型:
bytes
类型。这只是一个字节序列,Python不知道关于如何将其解释为字符的任何信息。str
类型。这也是一个字节序列,但是 Python 知道如何将这些字节解释为字符。unicode
类型已被删除。 str
现在支持 unicode。在 Python 2 中,隐式假设编码可能会导致很多问题;你最终可能会使用错误的编码,或者数据可能没有编码全部(例如,它是 PNG 图像)。
明确告诉 Python 使用哪种编码(或明确告诉它猜测)通常更好,更符合“Python哲学”的“explicit is better than implicit”。
此更改与 Python 2 不兼容,因为许多返回值已更改,导致像这样的微妙问题;这可能是主要原因Python 3 的采用非常缓慢。由于 Python 没有静态类型2不可能使用脚本自动更改它(例如捆绑的2to3
).
bytes('h€llo', 'utf-8')
将 str
转换为 bytes
;这应该生成 b'H\xe2\x82\xacllo'
。请注意如何将一个字符转换为三个字节。b'H\xe2\x82\xacllo'.decode('utf-8')
将 bytes
转换为 str
.当然,在您的情况下,UTF-8 可能不是正确的字符集,所以请确保使用正确的。
在您的特定代码中,nextline
的类型是 bytes
,而不是 str
,从 subprocess
读取 stdout
和 stdin
在 Python 3 中从 str
更改为字节
。这是因为 Python 无法确定它使用的是哪种编码。它可能使用与 sys.stdin.encoding
相同的(您系统的编码),但不能确定。
你需要更换:
sys.stdout.write(nextline)
与:
sys.stdout.write(nextline.decode('utf-8'))
或者也许:
sys.stdout.write(nextline.decode(sys.stdout.encoding))
您还需要将 if nextline == ''
修改为 if nextline == b''
因为:
>>> '' == b''
False
另见 Python 3 ChangeLog , PEP 358 , 和 PEP 3112 .
1 有一些巧妙的技巧可以用 ASCII 做,而用多字节字符集做不到;最著名的例子是“xor with space to switch case”(例如 chr(ord('a') ^ ord(' ')) == 'A'
)和“将第 6 位设置为制作一个控制字符”(例如 ord('\t') + ord('@') == ord('I')
)。 ASCII 是在操作单个位是一项对性能影响不可忽略的操作时设计的。
2是的,你可以使用函数注解,但这是一个比较新的特性,很少使用。
关于Python 3 TypeError : must be str, not bytes with sys.stdout.write(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21689365/
我有以下代码来捕获和处理运行命令输出。 如何修改它以便运行命令窗口显示输出并同时记录输出?更换 @SW_HIDE与 @SW_SHOW (或等效的)只显示一个空白的命令窗口。 类似于linux的东西te
[已编辑] 用于处理 subprocess.Popen 标准输出的 2 个选项是 stdout="a_file_name" 和 stdout=subprocess.PIPE。 stderr 可以通过
c99 的 7.19.3/7 节指出: At program start-up, three text streams are predefined and need not be opened ex
我正在运行以下 proc_open 函数。加载页面时,出现错误: Use of undefined constant STDOUT - assumed 'STDOUT'` 如何正确设置STDOUT和S
我有一个运行多个进程的开发堆栈:网络服务器、自动测试、后台编译等。所有这些都是基本的命令行命令,例如 node app.js 或 lein midje :autotest. 是否可以使用一个脚本在“后
我正在使用 SLURM 在 super 计算机上调度作业。我已设置 --output=log.out 选项,将作业标准输出中的内容放入文件 (log.out)。我发现该文件每 30-60 分钟更新一次
ansible/ansible-playbook 版本:2.1.2.0/2.2.0.0 我正在尝试使用 yum/apt 安装软件包,但由于安装软件包的存储库位于 packagecloud.io 中,有
bala@hp:~$ echo "Hello World" > stdout bala@hp:~$ cat stdout Hello World bala@hp:~$ echo "Hello Worl
在从 Fortran 编写的外部库中调用嘈杂的函数之前,我正在使用如下代码重定向标准输出: // copy standard output out = dup(STDOUT_FILENO); // c
这个问题在这里已经有了答案: How can I pipe stderr, and not stdout? (11 个答案) 关闭 6 年前。 我有一个程序,我想检查其 STDERR 输出并在其上运
我正在从 perl 运行一个 java 应用程序 这是脚本 #!/usr/bin/perl use strict; use warnings; $| = 1; my $args = join (" "
我正在尝试将 python 脚本的 STDOUT 重定向到一个文件。 如果 STDOUT 是从 sys 导入的,脚本的输出不会被重定向到一个文件: from sys import stdout std
我正在尝试使用 PHP 和 Apache(在 Docker 的前台运行)写入 stdout(或 stderr)。 这些作品: file_put_contents( "php://stderr","wo
我正在尝试重定向标准输出,以便 Windows 应用程序中的 printf 将转到我选择的文件。 我这样做: outFile = fopen("log.txt", "w"); *stdout = *o
在 Ruby 中,$stdout(前面有一个美元符号)和 STDOUT(全部大写)有什么区别?在进行输出重定向时,应该使用哪个,为什么? $stderr 和 STDERR 也是如此。 编辑: 刚找到一
让我们看看这个Hello World程序 #include int main(int argc, char ** argv) { printf("Hello, World!"); c
我在 64 位机器上运行 Ubuntu 20.04。 我想将 stdout 重定向到从 memfd_create 获得的描述符。似乎使用 memfd_create 创建的匿名文件只有在 stdout
我想在 Python 脚本末尾捕获控制台输出。也就是说,我既想正常打印到控制台,又想在执行结束时将控制台输出保存到文件中。 我看过各种相关的 SO 问题 1 , 2 , 3尽管他们要么简单地重定向输出
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我是一名优秀的程序员,十分优秀!