在控制台中重写多行

2024-12-31 08:38:00
admin
原创
126
摘要:问题描述:我知道可以用“\r”一致地重写终端上显示的最后一行,但我无法弄清楚是否有办法返回并编辑控制台中打印的先前的行。我想做的是重印多行用于基于文本的 RPG,然而,我的一位朋友也对此感到疑惑,因为有一个应用程序,其中一行专用于进度条,另一行描述下载。即控制台将打印:Moving file: NameOfF...

问题描述:

我知道可以用“\r”一致地重写终端上显示的最后一行,但我无法弄清楚是否有办法返回并编辑控制台中打印的先前的行。

我想做的是重印多行用于基于文本的 RPG,然而,我的一位朋友也对此感到疑惑,因为有一个应用程序,其中一行专用于进度条,另一行描述下载。

即控制台将打印:

Moving file: NameOfFile.txt  
Total Progress: [########              ] 40%

然后在程序运行时进行适当的更新(对两行)。


解决方案 1:

在 Unix 上,使用curses模块。

在 Windows 上,有几种选择:

使用 curses 的简单示例(我完全是 curses 新手):

import curses
import time

def report_progress(filename, progress):
    """progress: 0-10"""
    stdscr.addstr(0, 0, "Moving file: {0}".format(filename))
    stdscr.addstr(1, 0, "Total progress: [{1:10}] {0}%".format(progress * 10, "#" * progress))
    stdscr.refresh()

if __name__ == "__main__":
    stdscr = curses.initscr()
    curses.noecho()
    curses.cbreak()

    try:
        for i in range(10):
            report_progress("file_{0}.txt".format(i), i+1)
            time.sleep(0.5)
    finally:
        curses.echo()
        curses.nocbreak()
        curses.endwin()

解决方案 2:

像这样:

#!/usr/bin/env python

import sys
import time
from collections import deque

queue = deque([], 3)
for t in range(20):
    time.sleep(0.5)
    s = "update %d" % t
    for _ in range(len(queue)):
        sys.stdout.write("x1b[1Ax1b[2K") # move up cursor and delete whole line
    queue.append(s)
    for i in range(len(queue)):
        sys.stdout.write(queue[i] + "
") # reprint the lines

我在用 Go 编写的Jiri项目中发现了这一点。

更好的是:完成后删除所有行:

#!/usr/bin/env python

import sys
import time
from collections import deque

queue = deque([], 3)
t = 0
while True:
    time.sleep(0.5)
    if t <= 20:
        s = "update %d" % t
        t += 1
    else:
        s = None
    for _ in range(len(queue)):
        sys.stdout.write("x1b[1Ax1b[2K") # move up cursor and delete whole line
    if s != None:
        queue.append(s)
    else:
        queue.popleft()
    if len(queue) == 0:
        break
    for i in range(len(queue)):
        sys.stdout.write(queue[i] + "
") # reprint the lines

解决方案 3:

最终,如果您想操纵屏幕,您需要使用底层操作系统库,通常是:

  • Linux 或 OSX 上的 curses(或 terminfo/termcap 数据库跟踪的底层终端控制代码)

  • Windows 上的 win32 控制台 API。

如果您不介意坚持使用一个操作系统或乐于在 Windows 上安装第三方库,@codeape 的回答已经为您提供了许多选项。

但是,如果您想要一个可以简单地 pip 安装的跨平台解决方案,则可以使用asciimatics。作为开发此包的一部分,我必须解决环境之间的差异,以提供适用于 Linux、OSX 和 Windows 的单一 API。

对于进度条,您可以使用 BarChart 对象,如本演示中使用此代码所示。

解决方案 4:

这是一个适用于 Python 2/3 的 Python 模块,它可以用几行代码简单地解决这种情况;D

reprint - Python 2/3 的一个简单模块,用于在终端中打印和刷新多行输出内容

您可以简单地将该output实例视为普通dict实例list(取决于您使用的模式)。当您修改output实例中的内容时,终端中的输出将自动刷新 :D

根据您的需要,这里是代码:

from reprint import output
import time

if __name__ == "__main__":
    with output(output_type='dict') as output_lines:
        for i in range(10):
            output_lines['Moving file'] = "File_{}".format(i)
            for progress in range(100):
                output_lines['Total Progress'] = "[{done}{padding}] {percent}%".format(
                    done = "#" * int(progress/10),
                    padding = " " * (10 - int(progress/10)),
                    percent = progress
                    )
                time.sleep(0.05)

解决方案 5:

回车键可用于转到行首,ANSI 代码ESC A( "") 可用于向上一行。这在 Linux 上有效。它可以在 Windows 上工作,方法是使用colorama软件包启用 ANSI 代码:

import time
import sys
import colorama

colorama.init()

print("Line 1")
time.sleep(1)
print("Line 2")
time.sleep(1)
print("Line 3 (no eol)", end="")
sys.stdout.flush()
time.sleep(1)
print("
Line 3 the sequel")
time.sleep(1)
print("Line 3 the second sequel")
time.sleep(1)
print("Line 1 the sequel")
time.sleep(1)
print()  # skip two lines so that lines 2 and 3 don't get overwritten by the next console prompt
print()

输出:

> python3 multiline.py
Line 1 the sequel
Line 2
Line 3 the second sequel
>

在底层,colorama 大概可以使用 启用控制台虚拟终端序列SetConsoleMode

(也发布在这里:https: //stackoverflow.com/a/64360937/461834)

解决方案 6:

您可以尝试tqdm。

from time import sleep
from tqdm import tqdm
from tqdm import trange

files = [f'file_{i}' for i in range(10)]
desc_bar = tqdm(files, bar_format='{desc}')
prog_bar = trange(len(files), desc='Total Progress', ncols=50, ascii=' #',
                  bar_format='{desc}: [{bar}] {percentage:3.0f}%')

for f in desc_bar:
    desc_bar.set_description_str(f'Moving file: {f}')
    prog_bar.update(1)
    sleep(0.25)

在此处输入图片描述

还有嵌套进度条功能tqdm

from tqdm.auto import trange
from time import sleep

for i in trange(4, desc='1st loop'):
    for k in trange(50, desc='2rd loop', leave=False):
        sleep(0.01)

在此处输入图片描述

请注意,嵌套进度条tqdm存在一些已知问题:

  • 控制台一般:需要支持将光标移动到上一行。例如,IDLE、ConEmu和PyCharm(也包括此处、此处和此处)缺乏全面支持。

  • Windows:另外可能还需要 Python 模块colorama来确保嵌套的条形图停留在各自的行内。

对于 Python 中的嵌套进度条,Python 中的双进度条 - Stack Overflow有更多信息。

解决方案 7:

我发现了一个使用“magic_char”的简单解决方案。

magic_char = ''
multi_line = 'First
Second
Third'
ret_depth = magic_char * multi_line.count('
')
print('{}{}'.format(ret_depth, multi_line), end='', flush = True)
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用