欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python3讀取和處理超大文件的操作詳解

 更新時(shí)間:2024年04月19日 10:23:33   作者:rs勿忘初心  
在日常工作中,文件對象是我們常接觸到的可迭代類型之一,一般用?for?循環(huán)遍歷一個(gè)文件對象,可以逐行讀取它的內(nèi)容,但這種方式在碰到大文件時(shí),可能會出現(xiàn)一些奇怪的效率問題,所以本文給大家介紹了Python3讀取和處理超大文件的操作,需要的朋友可以參考下

需求:

小明是一位 Python 初學(xué)者,在學(xué)習(xí)了如何用 Python 讀取文件后,他想要做一個(gè)小練習(xí):計(jì)算某個(gè)文件中數(shù)字字符(0~9)的數(shù)量。

場景1:小文件處理

假設(shè)現(xiàn)在有一個(gè)測試用的小文件 small_file.txt,里面包含了一行行的隨機(jī)字符串:

feiowe9322nasd9233rl
aoeijfiowejf8322kaf9a
...

代碼示例:file_process.py

def count_digits(fname):
    """計(jì)算文件里包含多少個(gè)數(shù)字字符"""
    count = 0
   
     with open(fname) as file:
        for line in file:
            for s in line:
                if s.isdigit():
                    count += 1
    return count
 
 
fname = "./small_file.txt"
print(count_digits(fname))

運(yùn)行結(jié)果:

# 運(yùn)行腳本
python3 ./file_process.py
 
# 輸出結(jié)果
13

場景2:大文件處理

假設(shè)現(xiàn)在我們的大文件big_file.txt,大小有5G,且所有的文本都在一行。

大文件 big_file.txt

df2if283rkwefh... <剩余 5 GB 大小> ...

卻發(fā)現(xiàn)同樣的程序花費(fèi)了一分多鐘才給出結(jié)果,并且整個(gè)執(zhí)行過程耗光了筆記本電腦的全部 4G 內(nèi)存。

問題分析:

為什么同一份代碼用于大文件時(shí),效率就會變低這么多呢?原因就藏在小明讀取文件的方法里。

在代碼里所使用的文件讀取方式,可謂 Python 里的“標(biāo)準(zhǔn)做法”:首先用 with open (fine_name) 上下文管理器語法獲得一個(gè)文件對象,然后用 for 循環(huán)迭代它,逐行獲取文件里的內(nèi)容。為什么這種文件讀取方式會成為標(biāo)準(zhǔn)?這是因?yàn)樗袃蓚€(gè)好處:

(1) with 上下文管理器會自動(dòng)關(guān)閉文件描述符;

(2) 在迭代文件對象時(shí),內(nèi)容是一行一行返回的,不會占用太多內(nèi)存。

不過這套標(biāo)準(zhǔn)做法雖好,但不是沒有缺點(diǎn)。假如被讀取的文件里 根本就沒有任何換行符,那么上面列的第 (2) 個(gè)好處就不再成立。缺少換行符以后,程序遍歷文件對象時(shí)就不知道該何時(shí)中斷,最終只能一次性生成一個(gè)巨大的字符串對象,白白消耗大量時(shí)間和內(nèi)存。這就是 count_digits() 函數(shù)在處理 big_file.txt 時(shí)變得異常緩慢的原因。

要解決這個(gè)問題,我們需要把這種讀取文件的“標(biāo)準(zhǔn)做法”暫時(shí)放到一邊。

解決方法:

使用 while 循環(huán)加 read() 方法分塊讀取。

除了直接遍歷文件對象來逐行讀取文件內(nèi)容外,我們還可以調(diào)用更底層的 file.read() 方法。與直接用循環(huán)迭代文件對象不同,每次調(diào)用 file.read(chunk_size), 會馬上讀取從當(dāng)前游標(biāo)位置往后 chunk_size 大小的文件內(nèi)容,不必等待任何換行符出現(xiàn)。有了 file.read() 方法的幫助,優(yōu)化后的代碼:

def count_digits_v2(fname):
    """計(jì)算文件里包含多少個(gè)數(shù)字字符,每次讀取 8 KB"""
    count = 0
    block_size = 1024 * 8
    with open(fname) as file:
        while True:
            chunk = file.read(block_size)
            # 當(dāng)文件沒有更多內(nèi)容時(shí),read 調(diào)用將會返回空字符串 ''
            if not chunk:
                break
            for s in chunk:
                if s.isdigit():
                    count += 1
    return count
 
 
fname = "./big_file.txt"
print(count_digits_v2(fname))

在新函數(shù)中,我們使用了一個(gè) while 循環(huán)來讀取文件內(nèi)容,每次最多讀 8 KB,程序不再需要在內(nèi)存中拼接長達(dá)數(shù)吉字節(jié)的字符串,內(nèi)存占用會大幅降低。

(吉字節(jié)是一種數(shù)據(jù)存儲單位,通常用于表示大容量存儲設(shè)備的容量大小。它等于1024^3(1,073,741,824)字節(jié),或者1,024兆字節(jié)。在計(jì)算機(jī)領(lǐng)域,常用于描述大型文件、程序或數(shù)據(jù)集的大小,例如硬盤容量、內(nèi)存容量等。)

拓展:用Python讀取超大文件中的部分行

Python文件讀取一直是python的常見用法,通用方法是直接readlines加載所有行。
但是,對于超大文件(如100G的tsv),直接加載所有行會非常慢。

如果是想遍歷整個(gè)文件并處理每一行,其實(shí)并不需要一次加載所有行。

這里用迭代器的方法來讀取文件:

file_name = 'all_items.tsv'
start_line = 110000
end_line = 120000
with open(file_name) as f:
	for i in range(0, start_line):
        next(f)
    lines = [next(f) for i in range(start_line, end_line)]
print(len(lines))

對于大文件all_items.tsv,咱們只讀取某一個(gè)區(qū)間的行來進(jìn)行處理。先用迭代器滾動(dòng)到start_line的位置,再開始讀取。
如果咱們可以利用迭代器,把這個(gè)文件的行遍歷一遍(并對行進(jìn)行處理):

file_name = 'all_items.tsv'
def process_line(line):
    return line
with open(file_name) as f:
    while True:
        try:
            line = next(f)
            process_line(line)
        except StopIteration:
            break

用一個(gè)while循環(huán)就可以完成從頭到尾行的遍歷。

以上就是Python3讀取和處理超大文件的操作詳解的詳細(xì)內(nèi)容,更多關(guān)于Python3讀取和處理超大文件的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • python利用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行命令的方法

    python利用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行命令的方法

    下面小編就為大家?guī)硪黄猵ython利用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行命令的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-10-10
  • Python辦公自動(dòng)化Word轉(zhuǎn)Excel文件批量處理

    Python辦公自動(dòng)化Word轉(zhuǎn)Excel文件批量處理

    這篇文章主要為大家介紹了Python辦公自動(dòng)化Word轉(zhuǎn)Excel文件批量處理示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06
  • Pandas封裝Excel工具類的方法步驟

    Pandas封裝Excel工具類的方法步驟

    本文主要介紹了Pandas封裝Excel工具類的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • 解決pyqt中ui編譯成窗體.py中文亂碼的問題

    解決pyqt中ui編譯成窗體.py中文亂碼的問題

    下面小編就為大家?guī)硪黄鉀Qpyqt中ui編譯成窗體.py中文亂碼的問題。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2016-12-12
  • Python Selenium常見的報(bào)錯(cuò)問題以及措施

    Python Selenium常見的報(bào)錯(cuò)問題以及措施

    這篇文章主要介紹了Python Selenium常見的報(bào)錯(cuò)問題以及措施,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • Python pyautogui模塊實(shí)現(xiàn)鼠標(biāo)鍵盤自動(dòng)化方法詳解

    Python pyautogui模塊實(shí)現(xiàn)鼠標(biāo)鍵盤自動(dòng)化方法詳解

    這篇文章主要介紹了Python pyautogui 模塊實(shí)現(xiàn)鼠標(biāo)鍵盤自動(dòng)化方法詳解,需要的朋友可以參考下
    2020-02-02
  • python有幾個(gè)版本

    python有幾個(gè)版本

    在本篇內(nèi)容里小編給大家分享的是關(guān)于python版本的相關(guān)知識點(diǎn)內(nèi)容,需要的朋友們可以學(xué)習(xí)下。
    2020-06-06
  • 跟老齊學(xué)Python之dict()的操作方法

    跟老齊學(xué)Python之dict()的操作方法

    這篇文章主要介紹了dict()的操作方法,由于dict的很多方法跟list類似,所以在詳細(xì)介紹dict的同時(shí),也跟list做了對比,是篇非常不錯(cuò)的文章,有需要的朋友參考下
    2014-09-09
  • 詳解python中的線程

    詳解python中的線程

    Python中創(chuàng)建線程有兩種方式:函數(shù)或者用類來創(chuàng)建線程對象。這篇文章主要介紹了python中的線程,需要的朋友可以參考下
    2018-02-02
  • Python編寫車票訂購系統(tǒng)?Python實(shí)現(xiàn)快遞收費(fèi)系統(tǒng)

    Python編寫車票訂購系統(tǒng)?Python實(shí)現(xiàn)快遞收費(fèi)系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了Python編寫車票訂購系統(tǒng),Python實(shí)現(xiàn)快遞收費(fèi)系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08

最新評論