Как читать большой текстовый файл построчно, используя Java?

736

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

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

  • 61
    @kamaci et. и др. Этот вопрос не должен быть помечен как дубликат. «Быстро прочитать последнюю строку» не является альтернативой, и ее спорно, является ли «быстрый способ чтения текстового файла строка за строкой». Самый быстрый способ сделать что-то не обязательно является обычным способом. Кроме того, ответы ниже включают код, а наиболее подходящая альтернатива, которую вы перечисляете, - нет. Этот вопрос полезен. В настоящее время это лучший результат поиска в Google по запросу "java read file by line". И, наконец, его устранение приводит к переполнению стека и обнаружению, что 1 в каждом 2 вопросе помечается для удаления.
  • 5
    Вот сравнение скорости для шести возможных реализаций.
Показать ещё 2 комментария
Теги:
performance
io
file-io
garbage-collection

21 ответ

978

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

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

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

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

try(BufferedReader br = new BufferedReader(new FileReader(file))) {
    for(String line; (line = br.readLine()) != null; ) {
        // process the line.
    }
    // line is not visible here.
}

ОБНОВЛЕНИЕ: в Java 8 вы можете сделать

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

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

  • 6
    Как выглядит этот шаблон при правильной обработке исключений? Я отмечаю, что br.close () генерирует IOException, что кажется удивительным - что может произойти при закрытии файла, открытого для чтения, в любом случае? Конструктор FileReader может вызвать исключение FileNotFound.
  • 0
    @MikeB В Java 7 вы бы добавили блок try-with-resource. Ожидается, что Close не вызовет исключение, но некоторые реализации могут. Я не ожидал бы, что BufferedReader - один из них. Очень редко разработчики добавляют специальную обработку для close();
Показать ещё 31 комментарий
127

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

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

// 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();
  • 5
    Мой файл 1,5 гигабайт, и невозможно прочитать файл, используя ваш ответ!
  • 3
    @AboozarRajabi Конечно, это возможно. Этот код может читать любой текстовый файл.
Показать ещё 1 комментарий
81

После отсутствует (март 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);
}
  • 1
    Используйте StandardCharsets.UTF_8 , используйте Stream<String> для краткости и избегайте использования forEach() и особенно forEachOrdered() если нет причины.
  • 2
    Зачем избегать forEach ()? Это плохо?
Показать ещё 3 комментария
34

Вот пример с полной обработкой ошибок и поддержкой спецификации кодировки для 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;
    }
}
  • 1
    Какое отношение ByteArrayInputStream строковым литералом, имеет к чтению большого текстового файла?
22

В 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 передает поток в Iterable.

  • 0
    Не реализуя Iterable этот код определенно уродлив, хотя и полезен. Для работы требуется приведение (то есть (Iterable<String>) ).
  • 0
    Как я могу пропустить первую строку с этим методом?
Показать ещё 4 комментария
19

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

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 - это текст типа файла.

  • 0
    Должно ли объявление метода выглядеть вместо этого: «public static void readText throws FileNotFoundException () {´ Like:« public static void readText () throws FileNotFoundException {´ »
  • 0
    Это значительно медленнее, чем BufferedReader.readLine() , и он попросил самый эффективный метод.
17

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), поэтому вам нужно указать кодировку.

  • 0
    Хорошая точка зрения. Другой способ - использовать Files.newBufferedReader
15

В 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);
}
  • 9
    быть в курсе! использование line.split таким способом НЕ будет анализироваться должным образом, если поле содержит запятую и окружено кавычками. Это разделение игнорирует это и просто разделяет поле на куски, используя внутреннюю запятую. HTH, Марсело.
  • 0
    CSV: файл значений, разделенных запятыми, поэтому вам не следует использовать запятую в поле csv, если только вы не хотите добавить другое поле. Таким образом, использовать разбивку для токена запятой в java при разборе CSV-файла совершенно нормально и правильно
Показать ещё 3 комментария
10

Я задокументировал и протестировал 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. Для считывания тестового файла объемом 1 ГБ потребовалось около 4,5 секунд.

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 методов чтения файлов здесь.

  • 1
    Ваш гид потрясающий :)
9

Для Чтение файла с помощью 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();
        }
        }
    }
9

Вы можете использовать класс Scanner

Scanner sc=new Scanner(file);
sc.nextLine();
  • 9
    Это будет ужасно бомбить большой файл. Большой файл был частью вопроса ОП.
  • 2
    @Tim «Бомба ужасно» - это не термин, который я узнаю в CS. Что именно ты имеешь ввиду?
Показать ещё 4 комментария
8

В Java 8 существует также альтернатива использованию Files.lines(). Если ваш источник ввода не является файлом, а чем-то более абстрактным, например Reader или InputStream, вы можете передавать строки с помощью метода BufferedReader lines().

Например:

try( BufferedReader reader = new BufferedReader( ... ) ) {
  reader.lines().foreach( line -> processLine( line ) );
}

вызовет processLine() для каждой строки ввода, прочитанной BufferedReader.

7

Вам нужно использовать метод readLine() в class BufferedReader. Создайте новый объект из этого класса и примените этот метод к нему и сохраните его в строке.

BufferReader Javadoc

  • 0
    Похоже, ссылка на BufferReaderAPI не работает
6

Java-9:

try (Stream<String> stream = Files.lines(Paths.get(fileName))) {
        stream.forEach(System.out::println);
}
  • 1
    У этого кода есть цель? Например, это быстрее?
  • 2
    Я думаю, что вы должны System.getProperty("os.name").equals("Linux")
Показать ещё 4 комментария
5

Ясный способ достичь этого,

Например:

Если у вас есть dataFile.txt в вашем текущем каталоге

import java.io.*;
import java.util.Scanner;
import java.io.FileNotFoundException;

public class readByLine
{
    public readByLine() throws FileNotFoundException
    {
        Scanner linReader = new Scanner(new File("dataFile.txt"));

        while (linReader.hasNext())
        {
            String line = linReader.nextLine();
            System.out.println(line);
        }
        linReader.close();

    }

    public static void main(String args[])  throws FileNotFoundException
    {
        new readByLine();
    }
}

Выход, как показано ниже, Изображение 1393

  • 0
    Почему это понятнее? И не размещайте фотографии текста здесь. Опубликовать текст.
  • 0
    Вы разместили картинку. Это изображение текста. Вы могли бы вырезать и вставить текст прямо на эту страницу. Никто ничего не сказал о размещении программ. Публикация изображений текста - это пустая трата вашего времени, которое мне безразлично, и пропускная способность, которую я делаю.
Показать ещё 1 комментарий
3
BufferedReader br;
FileInputStream fin;
try {
    fin = new FileInputStream(fileName);
    br = new BufferedReader(new InputStreamReader(fin));

    /*Path pathToFile = Paths.get(fileName);
    br = Files.newBufferedReader(pathToFile,StandardCharsets.US_ASCII);*/

    String line = br.readLine();
    while (line != null) {
        String[] attributes = line.split(",");
        Movie movie = createMovie(attributes);
        movies.add(movie);
        line = br.readLine();
    }
    fin.close();
    br.close();
} catch (FileNotFoundException e) {
    System.out.println("Your Message");
} catch (IOException e) {
    System.out.println("Your Message");
}

Это работает для меня. Надеюсь, это тоже поможет.

2

Обычно я обычно выполняю процедуру чтения:

void readResource(InputStream source) throws IOException {
    BufferedReader stream = null;
    try {
        stream = new BufferedReader(new InputStreamReader(source));
        while (true) {
            String line = stream.readLine();
            if(line == null) {
                break;
            }
            //process line
            System.out.println(line)
        }
    } finally {
        closeQuiet(stream);
    }
}

static void closeQuiet(Closeable closeable) {
    if (closeable != null) {
        try {
            closeable.close();
        } catch (IOException ignore) {
        }
    }
}
0

Использование пакета org.apache.commons.io дало большую производительность, особенно в устаревшем коде, который использует Java 6 и ниже.
Для Java 7 он имеет лучший API с меньшим количеством обработок исключений и более полным использованием полных методов

LineIterator lineIterator =null;
    try{
    lineIterator = FileUtils.lineIterator(new File("/home/username/m.log"), "windows-1256");//second parameter is optionanl
    while (lineIterator.hasNext()){
      String currentLine = lineIterator.next();   
     //some operation
    } 
    }finally {  
     LineIterator.closeQuietly(lineIterator);
    }

специалист

<!-- https://mvnrepository.com/artifact/commons-io/commons-io -->
<dependency>
    <groupId>commons-io</groupId>
    <artifactId>commons-io</artifactId>
    <version>2.6</version>
</dependency>
0

Вы можете использовать этот код:

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

public class ReadTextFile {

    public static void main(String[] args) throws IOException {

        try {

            File f = new File("src/com/data.txt");

            BufferedReader b = new BufferedReader(new FileReader(f));

            String readLine = "";

            System.out.println("Reading file using Buffered Reader");

            while ((readLine = b.readLine()) != null) {
                System.out.println(readLine);
            }

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

    }

}
0

Вы можете использовать потоки, чтобы сделать это более точно:

Files.lines(Paths.get("input.txt")).forEach(s -> stringBuffer.append(s);
  • 1
    Что не так в этом ответе?
  • 1
    Что здесь не так ??? Почему никто не дает комментарии ??
Показать ещё 1 комментарий
0

Вы также можете использовать apache commons io:

File file = new File("/home/user/file.txt");
try {
    List<String> lines = FileUtils.readLines(file);
} catch (IOException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
}
  • 3
    FileUtils.readLines(file) является устаревшим методом. Кроме того, метод вызывает IOUtils.readLines , который использует BufferedReader и ArrayList. Это не построчный метод, и, конечно, он не подходит для чтения нескольких ГБ.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню