为什么 print() 和记录器对象以意想不到的顺序流式传输到控制台输出?
Why does print() and logger object streaming to console outputs in unexpected order?
我尝试在使用两个记录器对象之前和之后使用打印功能,这两个记录器对象会将输出流式传输到控制台。
事情是我得到了一个非常混乱的输出,这不是预期的顺序。
我试过将 print() 更改为 logger_object.info() 并且效果很好。
带有意外输出的完整代码:
import logging
# Create logger objects and set level
loggerA = logging.getLogger()
loggerA.setLevel(logging.DEBUG)
loggerB = logging.getLogger()
loggerB.setLevel(logging.DEBUG)
# Set log output format
log_format = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s', datefmt='%m/%d/%Y - %I:%M:%S %p')
# Create a console stream handler
stream_handler = logging.StreamHandler()
stream_handler.setFormatter(log_format)
# Add handlers to the loggers
loggerA.addHandler(stream_handler)
loggerB.addHandler(stream_handler)
def main():
for i in range(10):
print('Starting loop number {}'.format(i))
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
print('Finished loop number {}'.format(i))
if __name__ == '__main__':
main()
添加 loggerC 对象以用 loggerC.info() 替换 print() :
...
loggerC = logging.getLogger()
loggerC.setLevel(logging.DEBUG)
...
def main():
for i in range(10):
loggerC('Starting loop number {}'.format(i))
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
loggerC('Finished loop number {}'.format(i))
...
第一个使用 print() 意外输出的情况是:
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 1
Starting loop number 1
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 1
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 2
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 2
Finished loop number 1
Starting loop number 2
Finished loop number 2
Starting loop number 3
Finished loop number 3
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 3
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 3
第二种情况使用 loggerC.info() 而不是预期输出的 print() :
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 3
您的问题可能是 standard streams. By default print
uses std::out
and logger.info
uses std::err
. Both streams are bound to your terminal 的问题,但它们可能有不同的 flush
刷新或触发器。
这就是为什么您的输出混合在一起,函数写入不同的流并且它们以不同的方式刷新到终端,从而导致明显不一致的结果。
更改您的第一个代码以强制 logger.info
使用与 print
相同的流解决您的问题:
import sys
stream_handler = logging.StreamHandler(sys.stdout)
现在,它 returns 正确的输出:
-- Starting loop number 0
04/06/2019 - 07:41:34 AM - INFO - root - loop number 0
04/06/2019 - 07:41:34 AM - INFO - root - loop number 0
-- Finished loop number 0
-- Starting loop number 1
04/06/2019 - 07:41:34 AM - INFO - root - loop number 1
04/06/2019 - 07:41:34 AM - INFO - root - loop number 1
-- Finished loop number 1
-- Starting loop number 2
04/06/2019 - 07:41:34 AM - INFO - root - loop number 2
04/06/2019 - 07:41:34 AM - INFO - root - loop number 2
-- Finished loop number 2
-- Starting loop number 3
04/06/2019 - 07:41:34 AM - INFO - root - loop number 3
04/06/2019 - 07:41:34 AM - INFO - root - loop number 3
-- Finished loop number 3
因为 print
和 logger.info
每次刷新到您的终端时都以正确的顺序提供相同的流,所以结果是正确的。
你也可以保留different streams and force the std::out
stream to be flushed explicitly:
def main():
for i in range(10):
print('-- Starting loop number {}'.format(i))
# Force std::out stream (fed by print) to be flushed to the terminal
# before logger feeds std::err and also flushes
sys.stdout.flush()
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
print('-- Finished loop number {}'.format(i))
sys.stdout.flush()
第二个版本也产生了预期的结果。
还要记住 logger
根据定义是 thread-safe 但 print
不是。如果您要使用线程制作模块,请仅使用 logger
来跟踪执行。
我尝试在使用两个记录器对象之前和之后使用打印功能,这两个记录器对象会将输出流式传输到控制台。 事情是我得到了一个非常混乱的输出,这不是预期的顺序。
我试过将 print() 更改为 logger_object.info() 并且效果很好。
带有意外输出的完整代码:
import logging
# Create logger objects and set level
loggerA = logging.getLogger()
loggerA.setLevel(logging.DEBUG)
loggerB = logging.getLogger()
loggerB.setLevel(logging.DEBUG)
# Set log output format
log_format = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s', datefmt='%m/%d/%Y - %I:%M:%S %p')
# Create a console stream handler
stream_handler = logging.StreamHandler()
stream_handler.setFormatter(log_format)
# Add handlers to the loggers
loggerA.addHandler(stream_handler)
loggerB.addHandler(stream_handler)
def main():
for i in range(10):
print('Starting loop number {}'.format(i))
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
print('Finished loop number {}'.format(i))
if __name__ == '__main__':
main()
添加 loggerC 对象以用 loggerC.info() 替换 print() :
...
loggerC = logging.getLogger()
loggerC.setLevel(logging.DEBUG)
...
def main():
for i in range(10):
loggerC('Starting loop number {}'.format(i))
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
loggerC('Finished loop number {}'.format(i))
...
第一个使用 print() 意外输出的情况是:
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 1
Starting loop number 1
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 1
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 2
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 2
Finished loop number 1
Starting loop number 2
Finished loop number 2
Starting loop number 3
Finished loop number 3
04/06/2019 - 10:10:34 AM - INFO - loggerA - loop number 3
04/06/2019 - 10:10:34 AM - INFO - loggerB - loop number 3
第二种情况使用 loggerC.info() 而不是预期输出的 print() :
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 1
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 2
04/06/2019 - 10:12:21 AM - INFO - loggerC - Starting loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerA - loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerB - loop number 3
04/06/2019 - 10:12:21 AM - INFO - loggerC - Finished loop number 3
您的问题可能是 standard streams. By default print
uses std::out
and logger.info
uses std::err
. Both streams are bound to your terminal 的问题,但它们可能有不同的 flush
刷新或触发器。
这就是为什么您的输出混合在一起,函数写入不同的流并且它们以不同的方式刷新到终端,从而导致明显不一致的结果。
更改您的第一个代码以强制 logger.info
使用与 print
相同的流解决您的问题:
import sys
stream_handler = logging.StreamHandler(sys.stdout)
现在,它 returns 正确的输出:
-- Starting loop number 0
04/06/2019 - 07:41:34 AM - INFO - root - loop number 0
04/06/2019 - 07:41:34 AM - INFO - root - loop number 0
-- Finished loop number 0
-- Starting loop number 1
04/06/2019 - 07:41:34 AM - INFO - root - loop number 1
04/06/2019 - 07:41:34 AM - INFO - root - loop number 1
-- Finished loop number 1
-- Starting loop number 2
04/06/2019 - 07:41:34 AM - INFO - root - loop number 2
04/06/2019 - 07:41:34 AM - INFO - root - loop number 2
-- Finished loop number 2
-- Starting loop number 3
04/06/2019 - 07:41:34 AM - INFO - root - loop number 3
04/06/2019 - 07:41:34 AM - INFO - root - loop number 3
-- Finished loop number 3
因为 print
和 logger.info
每次刷新到您的终端时都以正确的顺序提供相同的流,所以结果是正确的。
你也可以保留different streams and force the std::out
stream to be flushed explicitly:
def main():
for i in range(10):
print('-- Starting loop number {}'.format(i))
# Force std::out stream (fed by print) to be flushed to the terminal
# before logger feeds std::err and also flushes
sys.stdout.flush()
loggerA.info("loop number {}".format(i))
loggerB.info("loop number {}".format(i))
print('-- Finished loop number {}'.format(i))
sys.stdout.flush()
第二个版本也产生了预期的结果。
还要记住 logger
根据定义是 thread-safe 但 print
不是。如果您要使用线程制作模块,请仅使用 logger
来跟踪执行。