【操作系统】为什么在Windows系统上tqdm跟print函数两者不是同步输出的呢?

2024-02-02 08:20

本文主要是介绍【操作系统】为什么在Windows系统上tqdm跟print函数两者不是同步输出的呢?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 问题描述

今天在学习tqdm的时候,遇到一个问题,就是我发现trange的输出跟主线程的print函数不是同步输出的
我使用的测试代码如下:

import sysfrom tqdm import trangedef main():X = 100for i in trange(X):# for i in trange(X, file=sys.stdout):print(i)for j in range(X):print(j)k = j * iprint("Done🛠️")return 0if __name__ == '__main__':main()

控制台的输出如下:

# 输出信息很多,这里就只给出了最后几行信息
...
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
Done🛠️
100%|██████████| 100/100 [00:00<00:00, 3694.12it/s]Process finished with exit code 0

可以看到100%|██████████| 100/100 [00:00<00:00, 3694.12it/s]信息竟然是在Done🛠️之后才输出的,这让我们感觉很奇怪;

2 假设和猜测

(1)猜测:tqdm在其底层实现中是使用子线程进行进度条的打印输出的

也就是,进度条的打印输出是异步进行的,所以才会出现【100%|██████████| 100/100 [00:00<00:00,...]进度条信息在Done🛠️】之后输出的情况;

(2)猜测:tqdm的进度条刷新也是通过end='\r'的设置实现的

我们如此猜测,是因为如果在中途进行print输出的话,就会重复显示进度条;我们使用了如下代码来进行测试:

import sys
from tqdm import trangedef main():X = 10k = 0for i in trange(X, file=sys.stderr):  # 使用默认的stderr输出print(f"Outer loop {i}")for j in range(2000000):k += j * iprint(f"Done🛠️: k = {k}")return 0if __name__ == '__main__':main()

其输出信息如下:

  0%|          | 0/10 [00:00<?, ?it/s]Outer loop 0
Outer loop 120%|██        | 2/10 [00:00<00:00, 11.00it/s]Outer loop 2
Outer loop 340%|████      | 4/10 [00:00<00:00, 10.02it/s]Outer loop 4
Outer loop 560%|██████    | 6/10 [00:00<00:00,  9.67it/s]Outer loop 670%|███████   | 7/10 [00:00<00:00,  9.52it/s]Outer loop 7
Outer loop 890%|█████████ | 9/10 [00:00<00:00,  9.32it/s]Outer loop 9
100%|██████████| 10/10 [00:01<00:00,  9.53it/s]
Done🛠️: k = 89999955000000Process finished with exit code 0

可以看到中途会出现循环过程中的进度条信息;

3 提问备忘

【StackOverflow】Understanding Threading Behavior in tqdm for Progress Bar Updates

这篇关于【操作系统】为什么在Windows系统上tqdm跟print函数两者不是同步输出的呢?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/670073

相关文章

Django中的函数视图和类视图以及路由的定义方式

《Django中的函数视图和类视图以及路由的定义方式》Django视图分函数视图和类视图,前者用函数处理请求,后者继承View类定义方法,路由使用path()、re_path()或url(),通过in... 目录函数视图类视图路由总路由函数视图的路由类视图定义路由总结Django允许接收的请求方法http

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

Linux系统之lvcreate命令使用解读

《Linux系统之lvcreate命令使用解读》lvcreate是LVM中创建逻辑卷的核心命令,支持线性、条带化、RAID、镜像、快照、瘦池和缓存池等多种类型,实现灵活存储资源管理,需注意空间分配、R... 目录lvcreate命令详解一、命令概述二、语法格式三、核心功能四、选项详解五、使用示例1. 创建逻

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

MySQL字符串常用函数详解

《MySQL字符串常用函数详解》本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql字符串常用函数一、获取二、大小写转换三、拼接四、截取五、比较、反转、替换六、去空白、填充MySQL字符串常用函数一、

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处