末尾没有换行符时,为什么打印输出没有立即显示在终端中?

问题描述 投票:0回答:1

我有一个执行模拟的 python 脚本。每次迭代都需要相当长的不同时间,所以我在每次循环后打印一个

.
作为一种方式来监视它运行的速度以及它在脚本运行时通过
for
语句的距离。所以代码有这样的一般结构:

for step in steps:
    run_simulation(step)
    # Python 3.x version:
    print('.', end='')
    # for Python 2.x:
    # print '.',

但是,当我运行代码时,点并没有一个一个地出现。相反,当循环结束时,所有的点都被立即打印出来,这使得整个工作变得毫无意义。如何在代码运行时内联打印点?


当迭代从另一个进程提供的数据并尝试打印结果时,例如回显来自 Electron 应用程序的输入,也会出现此问题。请参阅Python 不打印输出

python terminal printing output-buffering
1个回答
19
投票

问题

默认情况下,Python 程序的输出是缓冲 以提高性能。终端是一个与您的代码独立的程序,存储文本并同时进行通信更有效,而不是分别要求终端程序显示每个符号。

由于终端程序通常是交互式使用的,输入和输出一次一行(例如,用户应该按下 Enter 以指示单个输入项的结束),默认情况下缓冲一次输出一行。

所以,如果没有换行

print
ed,
print
函数(在 3.x 中;
print
语句在 2.x 中)将简单地向缓冲区添加文本,并且不显示任何内容。

其他方式输出

时不时地,有人会尝试直接使用标准输出流从 Python 程序中输出:

import sys
sys.stdout.write('test')

这会遇到同样的问题:如果输出不以换行符结尾,它将位于缓冲区中直到被刷新。

解决问题

单身
print

我们可以显式地在打印后刷新输出

在 3.x 中,

print
函数有一个
flush
关键字参数,可以直接解决问题:

for _ in range(10):
    print('.', end=' ', flush=True)
    time.sleep(.2)  # or other time-consuming work

在 2.x 中,

print
语句不提供此功能。相反,使用其
.flush
方法显式刷新流。标准输出流(默认情况下,
print
ed 时文本所在的位置)由
sys
标准库模块提供,并命名为
stdout
。因此,代码将如下所示:

for _ in range(10):
    print '.',
    sys.stdout.flush()
    time.sleep(.2)  # or other time-consuming work

对于多个
print
s

不是在每

print
之后冲洗一次(或决定哪些需要之后冲洗),可以完全禁用输出线缓冲。有很多方法可以做到这一点,所以请参考链接的问题。

© www.soinside.com 2019 - 2024. All rights reserved.