如何使用Java逐行读取大型文本文件?

我需要使用Java逐行读取大约5-6 GB的大文本文件。

我怎样才能快速做到这一点?

1419381 次浏览

看看这个博客:

可以指定缓冲区大小,或可以使用默认大小。该默认值对于大多数目的。

// Open the fileFileInputStream fstream = new FileInputStream("textfile.txt");BufferedReader br = new BufferedReader(new InputStreamReader(fstream));
String strLine;
//Read File Line By Linewhile ((strLine = br.readLine()) != null)   {// Print the content on the consoleSystem.out.println (strLine);}
//Close the input streamfstream.close();

您需要在class BufferedReader中使用readLine()方法。从该类创建一个新对象并对他操作此方法并将其保存为字符串。

BufferReader Javadoc

您可以使用Scanner类

Scanner sc=new Scanner(file);sc.nextLine();

一个常见的模式是使用

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.}

更新:在Java8你可以做

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

注意:您必须将Stream放置在资源试用块中,以确保在其上调用#关闭方法,否则底层文件句柄永远不会关闭,直到GC稍后才会关闭。

这是一个包含完整错误处理和支持预Java7字符集规范的示例。对于Java7,您可以使用try-with-资源语法,这使代码更清晰。

如果您只是想要默认字符集,您可以跳过InputStream并使用FileReader。

InputStream ins = null; // raw byte-streamReader r = null; // cooked readerBufferedReader br = null; // buffered for readLine()try {String s;if (true) {String data = "#foobar\t1234\n#xyz\t5678\none\ttwo\n";ins = new ByteArrayInputStream(data.getBytes());} else {ins = new FileInputStream("textfile.txt");}r = new InputStreamReader(ins, "UTF-8"); // leave charset out for defaultbr = 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;}}

一旦Java8发布(2014年3月),您将能够使用流:

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);}

在Java8中,你可以这样做:

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。

Java7:

String folderPath = "C:/folderOfMyFile";Path path = Paths.get(folderPath, "myFileName.csv"); //or any text file eg.: txt, bat, etcCharset charset = Charset.forName("UTF-8");
try (BufferedReader reader = Files.newBufferedReader(path , charset)) {while ((line = reader.readLine()) != null ) {//separate all csv fields into string arrayString[] lineVariables = line.split(",");}} catch (IOException e) {System.err.println(e);}

Java9:

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

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),因此您必须指定编码。

您也可以使用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 blocke.printStackTrace();}

我通常简单地做阅读例程:

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 lineSystem.out.println(line)}} finally {closeQuiet(stream);}}
static void closeQuiet(Closeable closeable) {if (closeable != null) {try {closeable.close();} catch (IOException ignore) {}}}

在Java8中,还有一种方法可以替代Files.lines()。如果您的输入源不是文件,而是更抽象的文件,如ReaderInputStream,您可以通过BufferedReaderslines()方法行。

例如:

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

将为BufferedReader读取的每个输入行调用processLine()

您可以使用扫描仪扫描整个文本并逐行浏览文本。当然,您应该导入以下内容:

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}}

扫描仪基本上是扫描所有文本。而同时循环用于遍历整个文本。

.hasNextLine()函数是一个布尔值,如果文本中还有更多行,则返回true。.nextLine()函数为您提供整行作为字符串,然后您可以按照自己想要的方式使用。尝试System.out.println(line)打印文本。

附注:. txt是文件类型text。

实现这一目标的明确途径,

例如:

如果您的当前目录上有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();}}

输出如下,输入图片描述

对于读取文件和Java8

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 blocke.printStackTrace();}}}
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");}

它对我有用。希望它也能帮助你。

您可以使用流更精确地执行此操作:

Files.lines(Paths.get("input.txt")).forEach(s -> stringBuffer.append(s);

您可以使用此代码:

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();}
}
}

我记录并测试了在Java中读取文件的10种不同方法,然后通过让它们读取1KB到1GB的测试文件来相互运行它们。以下是读取1GB测试文件的最快3种文件读取方法。

请注意,在运行性能测试时,我没有向控制台输出任何内容,因为这会减慢测试速度。我只是想测试原始读取速度。

java.nio.file.Files.readAllBytes()

在Java7,8,9中测试。这是总体上最快的方法。读取1GB文件始终不到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);}}}

java.nio.file.Files.lines()

这在Java8和9中测试成功,但由于缺乏对lambda表达式的支持,它在Java7中不起作用。读取1GB文件大约需要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

测试工作在Java7,8,9。这花了大约4.5秒读取一个1GB的测试文件。

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种文件阅读方法这里的完整排名。

通过使用org.apache.commons.io包,它提供了更多的性能,特别是在使用Java6及以下的遗留代码中。

Java7有更好的API和更少的异常更实用的方法:

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

Maven

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

您可以逐行读取文件数据,如下所示:

String fileLoc = "fileLocationInTheDisk";
List<String> lines = Files.lines(Path.of(fileLoc), StandardCharsets.UTF_8).collect(Collectors.toList());