Как читать большой текстовый файл построчно, используя Java? - PullRequest
785 голосов
/ 03 мая 2011

Мне нужно построчно читать большой текстовый файл размером около 5-6 ГБ, используя Java.

Как я могу сделать это быстро?

Ответы [ 21 ]

996 голосов
/ 03 мая 2011

Обычным примером является использование

try (BufferedReader br = new BufferedReader(new FileReader(file))) {
    String line;
    while ((line = br.readLine()) != null) {
       // process the line.
    }
}

Вы можете читать данные быстрее, если предположите, что кодировка символов отсутствует.например, ASCII-7, но это не будет иметь большого значения.Весьма вероятно, что то, что вы делаете с данными, займет гораздо больше времени.

РЕДАКТИРОВАТЬ: менее распространенный шаблон для использования, который позволяет избежать охвата line утечки.1010 * ОБНОВЛЕНИЕ: В Java 8 вы можете выполнить

try (Stream<String> stream = Files.lines(Paths.get(fileName))) {
        stream.forEach(System.out::println);
}

ПРИМЕЧАНИЕ. Необходимо поместить поток в блок try-with-resource , чтобы обеспечить вызов метода #closeэто, иначе основной дескриптор файла никогда не будет закрыт, пока GC не сделает это намного позже.

138 голосов
/ 03 мая 2011

Посмотрите этот блог:

Размер буфера может быть указанили можно использовать размер по умолчанию.Значение по умолчанию достаточно велико для большинства целей.

// Open the file
FileInputStream fstream = new FileInputStream("textfile.txt");
BufferedReader br = new BufferedReader(new InputStreamReader(fstream));

String strLine;

//Read File Line By Line
while ((strLine = br.readLine()) != null)   {
  // Print the content on the console
  System.out.println (strLine);
}

//Close the input stream
fstream.close();
85 голосов
/ 25 июля 2013

После выхода (март 2014 г.) вы сможете использовать потоки:

try (Stream<String> lines = Files.lines(Paths.get(filename), Charset.defaultCharset())) {
  lines.forEachOrdered(line -> process(line));
}

Печать всех строк в файле:

try (Stream<String> lines = Files.lines(file, Charset.defaultCharset())) {
  lines.forEachOrdered(System.out::println);
}
35 голосов
/ 27 марта 2013

Вот пример с полной обработкой ошибок и поддержкой спецификации кодировки для pre-Java 7. В Java 7 вы можете использовать синтаксис try-with-resources, который делает код чище.

Если вы просто хотитекодировка по умолчанию, вы можете пропустить InputStream и использовать FileReader.

InputStream ins = null; // raw byte-stream
Reader r = null; // cooked reader
BufferedReader br = null; // buffered for readLine()
try {
    String s;
    ins = new FileInputStream("textfile.txt");
    r = new InputStreamReader(ins, "UTF-8"); // leave charset out for default
    br = new BufferedReader(r);
    while ((s = br.readLine()) != null) {
        System.out.println(s);
    }
}
catch (Exception e)
{
    System.err.println(e.getMessage()); // handle exception
}
finally {
    if (br != null) { try { br.close(); } catch(Throwable t) { /* ensure close happens */ } }
    if (r != null) { try { r.close(); } catch(Throwable t) { /* ensure close happens */ } }
    if (ins != null) { try { ins.close(); } catch(Throwable t) { /* ensure close happens */ } }
}

Вот версия Groovy с полной обработкой ошибок:

File f = new File("textfile.txt");
f.withReader("UTF-8") { br ->
    br.eachLine { line ->
        println line;
    }
}
21 голосов
/ 15 декабря 2013

В Java 8 вы можете сделать:

try (Stream<String> lines = Files.lines (file, StandardCharsets.UTF_8))
{
    for (String line : (Iterable<String>) lines::iterator)
    {
        ;
    }
}

Некоторые примечания: поток, возвращаемый Files.lines (в отличие от большинства потоков), необходимо закрыть.По причинам, указанным здесь Я избегаю использования forEach().Странный код (Iterable<String>) lines::iterator приводит поток к итерируемому.

19 голосов
/ 12 сентября 2015

Что вы можете сделать, это отсканировать весь текст с помощью сканера и проходить по тексту построчно. Конечно, вы должны импортировать следующее:

import java.io.File;
import java.io.FileNotFoundException;
import java.util.Scanner;
public static void readText throws FileNotFoundException {
    Scanner scan = new Scanner(new File("samplefilename.txt"));
    while(scan.hasNextLine()){
        String line = scan.nextLine();
        //Here you can manipulate the string the way you want
    }
}

Сканер в основном сканирует весь текст. Цикл while используется для перемещения по всему тексту.

Функция .hasNextLine() является логическим значением, которое возвращает true, если в тексте еще больше строк. Функция .nextLine() дает вам целую строку в виде строки, которую вы можете использовать по своему усмотрению. Попробуйте System.out.println(line) напечатать текст.

Боковое примечание. .Txt - это тип файла text.

17 голосов
/ 26 января 2015

FileReader не позволит вам указать кодировку, вместо этого используйте InputStreamReader, если вам нужно ее указать:

try {
    BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream(filePath), "Cp1252"));         

    String line;
    while ((line = br.readLine()) != null) {
        // process the line.
    }
    br.close();

} catch (IOException e) {
    e.printStackTrace();
}

Если вы импортировали этот файл из Windows, он может иметь кодировку ANSI (Cp1252), поэтому вы должны указать кодировку.

16 голосов
/ 09 апреля 2014

В Java 7:

String folderPath = "C:/folderOfMyFile";
Path path = Paths.get(folderPath, "myFileName.csv"); //or any text file eg.: txt, bat, etc
Charset charset = Charset.forName("UTF-8");

try (BufferedReader reader = Files.newBufferedReader(path , charset)) {
  while ((line = reader.readLine()) != null ) {
    //separate all csv fields into string array
    String[] lineVariables = line.split(","); 
  }
} catch (IOException e) {
    System.err.println(e);
}
14 голосов
/ 08 апреля 2018

Я задокументировал и протестировал 10 различных способов чтения файлов в Java , а затем применил их друг к другу, заставив их читать в тестовых файлах от 1 КБ до 1 ГБ.Вот 3 самых быстрых метода чтения файлов для чтения тестового файла объемом 1 ГБ.

Обратите внимание, что при выполнении тестов производительности я ничего не выводил на консоль, поскольку это действительно замедляло бы тестирование.Я просто хотел проверить скорость чтения в исходном формате.

1) java.nio.file.Files.readAllBytes ()

Протестировано в Java 7, 8, 9. В целом это был самый быстрый метод,Чтение файла объемом 1 ГБ всегда было менее 1 секунды.

import java.io..File;
import java.io.IOException;
import java.nio.file.Files;

public class ReadFile_Files_ReadAllBytes {
  public static void main(String [] pArgs) throws IOException {
    String fileName = "c:\\temp\\sample-1GB.txt";
    File file = new File(fileName);

    byte [] fileBytes = Files.readAllBytes(file.toPath());
    char singleChar;
    for(byte b : fileBytes) {
      singleChar = (char) b;
      System.out.print(singleChar);
    }
  }
}

2) java.nio.file.Files.lines ()

Это было успешно протестировано в Java 8 и 9, но этоне будет работать в Java 7 из-за отсутствия поддержки лямбда-выражений.Чтение файла размером 1 ГБ заняло около 3,5 секунд, что ставит его на второе место после чтения больших файлов.

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.util.stream.Stream;

public class ReadFile_Files_Lines {
  public static void main(String[] pArgs) throws IOException {
    String fileName = "c:\\temp\\sample-1GB.txt";
    File file = new File(fileName);

    try (Stream linesStream = Files.lines(file.toPath())) {
      linesStream.forEach(line -> {
        System.out.println(line);
      });
    }
  }
}

3) BufferedReader

Протестировано для работы в Java 7, 8, 9. Это заняло около 4,5 секунд, чтобы прочитать в тестовом файле 1 ГБ.

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;

public class ReadFile_BufferedReader_ReadLine {
  public static void main(String [] args) throws IOException {
    String fileName = "c:\\temp\\sample-1GB.txt";
    FileReader fileReader = new FileReader(fileName);

    try (BufferedReader bufferedReader = new BufferedReader(fileReader)) {
      String line;
      while((line = bufferedReader.readLine()) != null) {
        System.out.println(line);
      }
    }
  }

Вы можете найти полный рейтинг для всех 10 методов чтения файлов здесь .

10 голосов
/ 20 апреля 2017

Для Чтение файла с Java 8

  package com.java.java8;

    import java.nio.file.Files;
    import java.nio.file.Paths;
    import java.util.stream.Stream;

    /**
     * The Class ReadLargeFile.
     *
     * @author Ankit Sood Apr 20, 2017
     */
    public class ReadLargeFile {

        /**
         * The main method.
         *
         * @param args
         *            the arguments
         */
        public static void main(String[] args) {
        try {
            Stream<String> stream = Files.lines(Paths.get("C:\\Users\\System\\Desktop\\demoData.txt"));
            stream.forEach(System.out::println);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        }
    }
...