Es gibt bereits viele gute Antworten, aber wenn Ihre gesamte Datei in einer einzigen Zeile steht und Sie trotzdem "Zeilen" (im Gegensatz zu Blöcken fester Größe) verarbeiten wollen, werden Ihnen diese Antworten nicht helfen.
In 99 % der Fälle ist es möglich, Dateien zeilenweise zu verarbeiten. Dann, wie in diesem Artikel vorgeschlagen Antwort können Sie das Dateiobjekt selbst als Lazy Generator verwenden:
with open('big.csv') as f:
for line in f:
process(line)
Es kann jedoch vorkommen, dass sehr große Dateien vorliegen, bei denen das Zeilentrennzeichen nicht '\n'
(ein häufiger Fall ist '|'
).
- Konvertieren
'|'
a '\n'
vor der Verarbeitung nicht möglich ist, weil dadurch Felder, die legitimerweise enthalten sind, durcheinander gebracht werden können '\n'
(z.B. freie Texteingabe durch den Benutzer).
- Die Verwendung der csv-Bibliothek scheidet auch deshalb aus, weil zumindest in frühen Versionen der lib, es ist fest programmiert, die Eingabe zeilenweise zu lesen .
Für diese Art von Situationen habe ich das folgende Snippet erstellt [Aktualisiert im Mai 2021 für Python 3.8+]:
def rows(f, chunksize=1024, sep='|'):
"""
Read a file where the row separator is '|' lazily.
Usage:
>>> with open('big.csv') as f:
>>> for r in rows(f):
>>> process(r)
"""
row = ''
while (chunk := f.read(chunksize)) != '': # End of file
while (i := chunk.find(sep)) != -1: # No separator found
yield row + chunk[:i]
chunk = chunk[i+1:]
row = ''
row += chunk
yield row
[Für ältere Versionen von Python]:
def rows(f, chunksize=1024, sep='|'):
"""
Read a file where the row separator is '|' lazily.
Usage:
>>> with open('big.csv') as f:
>>> for r in rows(f):
>>> process(r)
"""
curr_row = ''
while True:
chunk = f.read(chunksize)
if chunk == '': # End of file
yield curr_row
break
while True:
i = chunk.find(sep)
if i == -1:
break
yield curr_row + chunk[:i]
curr_row = ''
chunk = chunk[i+1:]
curr_row += chunk
Ich konnte es erfolgreich zur Lösung verschiedener Probleme einsetzen. Es wurde ausgiebig getestet, mit verschiedenen Chunk-Größen. Hier ist das Testpaket, das ich verwende, für diejenigen, die sich selbst überzeugen müssen:
test_file = 'test_file'
def cleanup(func):
def wrapper(*args, **kwargs):
func(*args, **kwargs)
os.unlink(test_file)
return wrapper
@cleanup
def test_empty(chunksize=1024):
with open(test_file, 'w') as f:
f.write('')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 1
@cleanup
def test_1_char_2_rows(chunksize=1024):
with open(test_file, 'w') as f:
f.write('|')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 2
@cleanup
def test_1_char(chunksize=1024):
with open(test_file, 'w') as f:
f.write('a')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 1
@cleanup
def test_1025_chars_1_row(chunksize=1024):
with open(test_file, 'w') as f:
for i in range(1025):
f.write('a')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 1
@cleanup
def test_1024_chars_2_rows(chunksize=1024):
with open(test_file, 'w') as f:
for i in range(1023):
f.write('a')
f.write('|')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 2
@cleanup
def test_1025_chars_1026_rows(chunksize=1024):
with open(test_file, 'w') as f:
for i in range(1025):
f.write('|')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 1026
@cleanup
def test_2048_chars_2_rows(chunksize=1024):
with open(test_file, 'w') as f:
for i in range(1022):
f.write('a')
f.write('|')
f.write('a')
# -- end of 1st chunk --
for i in range(1024):
f.write('a')
# -- end of 2nd chunk
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 2
@cleanup
def test_2049_chars_2_rows(chunksize=1024):
with open(test_file, 'w') as f:
for i in range(1022):
f.write('a')
f.write('|')
f.write('a')
# -- end of 1st chunk --
for i in range(1024):
f.write('a')
# -- end of 2nd chunk
f.write('a')
with open(test_file) as f:
assert len(list(rows(f, chunksize=chunksize))) == 2
if __name__ == '__main__':
for chunksize in [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024]:
test_empty(chunksize)
test_1_char_2_rows(chunksize)
test_1_char(chunksize)
test_1025_chars_1_row(chunksize)
test_1024_chars_2_rows(chunksize)
test_1025_chars_1026_rows(chunksize)
test_2048_chars_2_rows(chunksize)
test_2049_chars_2_rows(chunksize)