我有一个执行模拟的 python 脚本。每次迭代都需要相当长的不同时间,所以我在每次循环后打印一个
.
作为一种方式来监视它运行的速度以及它在脚本运行时通过 for
语句的距离。所以代码有这样的一般结构:
for step in steps:
run_simulation(step)
# Python 3.x version:
print('.', end='')
# for Python 2.x:
# print '.',
但是,当我运行代码时,点并没有一个一个地出现。相反,当循环结束时,所有的点都被立即打印出来,这使得整个工作变得毫无意义。如何在代码运行时内联打印点?
当迭代从另一个进程提供的数据并尝试打印结果时,例如回显来自 Electron 应用程序的输入,也会出现此问题。请参阅Python 不打印输出。
默认情况下,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
之后冲洗一次(或决定哪些需要之后冲洗),可以完全禁用输出线缓冲。有很多方法可以做到这一点,所以请参考链接的问题。