【问题标题】:Download progressbar for Python 3下载 Python 3 的进度条
【发布时间】:2012-12-14 15:00:17
【问题描述】:

我需要在 Python 3 的文件下载期间显示进度。 我在 Stackoverflow 上看过一些主题,但考虑到我是编程菜鸟,没有人发布完整的示例,只是其中的一小部分,或者我可以在 Python 3 上使用的示例,没有一个对我有好处.. .

附加信息:

好的,我有这个:

from urllib.request import urlopen
import configparser
#checks for files which need to be downloaded
print('    Downloading...')
file = urlopen(file_url)
#progress bar here
output = open('downloaded_file.py','wb')
output.write(file.read())
output.close()
os.system('downloaded_file.py')

脚本通过python命令行运行

【问题讨论】:

  • 这个问题缺少很多有用的信息:你想要这个进度条在哪里?在控制台中?在网页上?在某些桌面应用程序中?你是如何下载文件的?等等等等
  • (Python 2 中的 urllib2 与 Python 3 中的 urllib 差不多)
  • 我注意到 python 3 中的 urllib.request 是 urllib2 是 python 2...如果我错了请纠正我...
  • @Cœur 该链接很有用(对于那些需要 Python 2 的人)。如果您愿意,可以将链接移动到 cmets¶ Python 3 代码可以改进(因此阻止未来的答案是没有用的)。

标签: python python-3.x download progress-bar


【解决方案1】:

urlretrieve() 可以将 url 下载到文件中,并允许指定一个 reporthook 回调来报告进度:

#!/usr/bin/env python3
import sys
from urllib.request import urlretrieve

def reporthook(blocknum, blocksize, totalsize):
    readsofar = blocknum * blocksize
    if totalsize > 0:
        percent = readsofar * 1e2 / totalsize
        s = "\r%5.1f%% %*d / %d" % (
            percent, len(str(totalsize)), readsofar, totalsize)
        sys.stderr.write(s)
        if readsofar >= totalsize: # near the end
            sys.stderr.write("\n")
    else: # total size is unknown
        sys.stderr.write("read %d\n" % (readsofar,))

urlretrieve(url, 'downloaded_file.py', reporthook)

这是一个 GUI 进度条:

import sys
from threading import Event, Thread
from tkinter import Tk, ttk
from urllib.request import urlretrieve

def download(url, filename):
    root = progressbar = quit_id = None
    ready = Event()
    def reporthook(blocknum, blocksize, totalsize):
        nonlocal quit_id
        if blocknum == 0: # started downloading
            def guiloop():
                nonlocal root, progressbar
                root = Tk()
                root.withdraw() # hide
                progressbar = ttk.Progressbar(root, length=400)
                progressbar.grid()
                # show progress bar if the download takes more than .5 seconds
                root.after(500, root.deiconify)
                ready.set() # gui is ready
                root.mainloop()
            Thread(target=guiloop).start()
        ready.wait(1) # wait until gui is ready
        percent = blocknum * blocksize * 1e2 / totalsize # assume totalsize > 0
        if quit_id is None:
            root.title('%%%.0f %s' % (percent, filename,))
            progressbar['value'] = percent # report progress
            if percent >= 100:  # finishing download
                quit_id = root.after(0, root.destroy) # close GUI

    return urlretrieve(url, filename, reporthook)

download(url, 'downloaded_file.py')

在 Python 3.3 上,urlretrieve() 具有不同的 reporthook 接口 (see issue 16409)。要解决这个问题,您可以通过FancyURLopener访问之前的界面:

from urllib.request import FancyURLopener
urlretrieve = FancyURLopener().retrieve

要在同一线程中更新进度条,您可以内联 urlretrieve() 代码:

from tkinter import Tk, ttk
from urllib.request import urlopen

def download2(url, filename):
    response = urlopen(url)
    totalsize = int(response.headers['Content-Length']) # assume correct header
    outputfile = open(filename, 'wb')

    def download_chunk(readsofar=0, chunksize=1 << 13):
        # report progress
        percent = readsofar * 1e2 / totalsize # assume totalsize > 0
        root.title('%%%.0f %s' % (percent, filename,))
        progressbar['value'] = percent

        # download chunk
        data = response.read(chunksize)
        if not data: # finished downloading
            outputfile.close()
            root.destroy() # close GUI
        else:
            outputfile.write(data) # save to filename
            # schedule to download the next chunk
            root.after(0, download_chunk, readsofar + len(data), chunksize)

    # setup GUI to show progress
    root = Tk()
    root.withdraw() # hide
    progressbar = ttk.Progressbar(root, length=400)
    progressbar.grid()
    # show progress bar if the download takes more than .5 seconds
    root.after(500, root.deiconify)
    root.after(0, download_chunk)
    root.mainloop()

download2(url, 'downloaded_file.py')

【讨论】:

  • @Mirac7:我添加了显示 GUI 的变体并从同一线程中下载文件。
  • 好的和完整的答案+1
  • @J.F.Sebastian 有没有办法在reporthook 中包含下载速度?
  • @GughanRavikumar 是的。存储你收到最后一个块的时间,然后除以一个块大小,得到当前的下载速度。
  • @GughanRavikumar 它可能看起来像:t = time.monotonic(); speed = blocksize / (t - last[0]); average_speed = readsofar / (t - start); last[0] = t
【解决方案2】:

我认为这段代码可以帮助你。我不太确定这正是你想要的。至少它应该给你一些工作。

import tkinter 
from tkinter import ttk
from urllib.request import urlopen


def download(event):
    file = urlopen('http://www.python.org/')
    output = open('downloaded_file.txt', 'wb')
    lines= file.readlines()
    i = len(lines)

    for line in lines:
        output.write(line)
        pbar.step(100/i)

    output.close()
    file.close()




root = tkinter.Tk()
root.title('Download bar')

pbar = ttk.Progressbar(root, length=300)
pbar.pack(padx=5, pady=5)

btn = tkinter.Button(root, text="Download")
# bind to left mouse button click
btn.bind("<Button-1>", download)
btn.pack(pady=10)

root.mainloop()

这个可行,我试过了。

【讨论】:

  • 我收到错误 io.UnsupportedOperation: seek... 我做错了吗?
  • 我认为 seek() 方法不适用于 urlopen()。删除行 file.seek(0) 并查看结果。
  • 已经试过了...它工作正常,但是当我点击下载时,它只是创建空脚本,并且进度条是空的。
  • @doru 为了避免打开 URL 两次,为什么不将 file.readlines() 分配给一个变量呢?
  • @poorsod 你是对的!非常感谢!我已经(重新)编辑了答案。为此 +1。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2018-01-08
  • 2013-03-16
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多