ฉันต้องอ่านไฟล์ข้อความขนาดใหญ่ประมาณ 5-6 GB ต่อบรรทัดโดยใช้ Java
ฉันจะทำสิ่งนี้ได้อย่างรวดเร็ว?
ฉันต้องอ่านไฟล์ข้อความขนาดใหญ่ประมาณ 5-6 GB ต่อบรรทัดโดยใช้ Java
ฉันจะทำสิ่งนี้ได้อย่างรวดเร็ว?
คำตอบ:
รูปแบบทั่วไปคือการใช้
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);
}
หมายเหตุ: คุณต้องวางสตรีมในบล็อกtry-with-resourceเพื่อให้แน่ใจว่ามีการเรียกเมธอด #close มิฉะนั้นจะไม่มีการปิดตัวจัดการไฟล์อ้างอิงจนกว่า GC จะทำในภายหลัง
for(String line = br.readLine(); line != null; line = br.readLine())
Btw ใน Java 8 คุณสามารถทำได้try( Stream<String> lines = Files.lines(...) ){ for( String line : (Iterable<String>) lines::iterator ) { ... } }
ซึ่งยากที่จะไม่เกลียด
ดูบล็อกนี้:
อาจระบุขนาดบัฟเฟอร์หรืออาจใช้ขนาดเริ่มต้น ค่าเริ่มต้นมีขนาดใหญ่พอสำหรับวัตถุประสงค์ส่วนใหญ่
// 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();
DataInputStream
และกระแสที่ไม่ถูกต้องถูกปิด ไม่มีอะไรผิดปกติกับ Java Tutorial และไม่จำเป็นต้องอ้างถึงอินเทอร์เน็ตของบุคคลที่สามที่ไม่ชอบอะไรเช่นนี้
เมื่อ Java 8 ออกมา (มีนาคม 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);
}
StandardCharsets.UTF_8
ใช้Stream<String>
เพื่อความรัดกุมและหลีกเลี่ยงการใช้forEach()
และโดยเฉพาะอย่างยิ่งforEachOrdered()
ถ้าไม่มีเหตุผล
forEach(this::process)
lambdas forEach()
forEachOrdered
ดำเนินการตามคำสั่ง โปรดระวังว่าคุณจะไม่สามารถขนานกระแสข้อมูลในกรณีนั้นแม้ว่าฉันจะพบว่าการขนานไม่เปิดเว้นแต่ว่าไฟล์จะมีหลายพันบรรทัด
นี่คือตัวอย่างที่มีการจัดการข้อผิดพลาดแบบเต็มและรองรับข้อมูลจำเพาะชุดอักขระสำหรับ pre-Java 7 ด้วย Java 7 คุณสามารถใช้ไวยากรณ์ลองกับทรัพยากรซึ่งทำให้โค้ดสะอาดขึ้น
หากคุณต้องการชุดอักขระเริ่มต้นคุณสามารถข้าม 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;
}
}
ByteArrayInputStream
fed โดยสตริงที่แท้จริงเกี่ยวข้องกับการอ่านไฟล์ข้อความขนาดใหญ่?
ใน 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
Iterable
รหัสนี้จะน่าเกลียดอย่างแน่นอนแม้ว่าจะมีประโยชน์ มันต้องการนักแสดง (เช่น(Iterable<String>)
) ในการทำงาน
for(String line : (Iterable<String>) lines.skip(1)::iterator)
Stream
คุณสมบัติจริง ๆให้ใช้Files.newBufferedReader
แทนFiles.lines
และโทรซ้ำ ๆreadLine()
จนกระทั่งnull
แทนที่จะใช้สิ่งก่อสร้าง(Iterable<String>) lines::iterator
ดูเหมือนว่าจะง่ายกว่ามาก ...
สิ่งที่คุณสามารถทำได้คือสแกนข้อความทั้งหมดโดยใช้เครื่องสแกนเนอร์และทำตามบรรทัดข้อความต่อบรรทัด แน่นอนคุณควรนำเข้าสิ่งต่อไปนี้:
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()
ฟังก์ชั่นบูลที่ส่งกลับจริงถ้ามีสายยังคงมากขึ้นในข้อความ .nextLine()
ฟังก์ชั่นช่วยให้คุณทั้งเส้นเป็นสตริงซึ่งคุณสามารถใช้วิธีการที่คุณต้องการ ลองSystem.out.println(line)
พิมพ์ข้อความ
หมายเหตุด้านข้าง: .txt เป็นข้อความประเภทไฟล์
BufferedReader.readLine()
มากและเขาขอวิธีที่ดีที่สุด
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) ดังนั้นคุณต้องระบุการเข้ารหัส
ฉันบันทึกและทดสอบ10 วิธีที่แตกต่างกันในการอ่านไฟล์ใน Javaและจากนั้นวิ่งต่อกันโดยทำให้พวกเขาอ่านในไฟล์ทดสอบจาก 1KB ถึง 1GB ต่อไปนี้เป็นวิธีการอ่านไฟล์ 3 วิธีที่เร็วที่สุดสำหรับการอ่านไฟล์ทดสอบ 1GB
โปรดทราบว่าเมื่อทำการทดสอบประสิทธิภาพฉันไม่ได้ส่งอะไรไปยังคอนโซลเนื่องจากจะทำให้การทดสอบช้าลงจริง ๆ ฉันแค่ต้องการทดสอบความเร็วในการอ่านแบบดิบ
1) java.nio.file.Files.readAllBytes ()
ทดสอบใน Java 7, 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);
}
}
}
2) java.nio.file.Files.lines ()
สิ่งนี้ถูกทดสอบเรียบร้อยแล้วใน Java 8 และ 9 แต่มันจะไม่ทำงานใน Java 7 เพราะขาดการสนับสนุนแลมบ์ดานิพจน์ ใช้เวลาประมาณ 3.5 วินาทีในการอ่านไฟล์ 1GB ซึ่งวางไว้ในตำแหน่งที่สองเท่าที่อ่านไฟล์ขนาดใหญ่
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 วินาทีในการอ่านไฟล์ทดสอบ 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 การอ่านไฟล์ที่นี่
System.out.print/println()
ที่นี่ คุณกำลังสมมติว่าไฟล์จะพอดีกับหน่วยความจำในสองกรณีแรกของคุณ
ใน 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);
}
StandardCharsets.UTF_8
เพื่อหลีกเลี่ยงข้อยกเว้นที่ตรวจสอบในCharset.forName("UTF-8")
ใน Java 8 Files.lines()
นอกจากนี้ยังมีทางเลือกที่จะใช้ หากแหล่งอินพุตของคุณไม่ใช่ไฟล์ แต่มีอะไรที่เป็นนามธรรมมากกว่าเช่นReader
หรือInputStream
คุณสามารถสตรีมบรรทัดผ่านเมธอดBufferedReader
slines()
ตัวอย่างเช่น:
try (BufferedReader reader = new BufferedReader(...)) {
reader.lines().forEach(line -> processLine(line));
}
จะเรียกสำหรับสายการป้อนข้อมูลแต่ละอ่านได้โดยprocessLine()
BufferedReader
สำหรับการอ่านไฟล์ด้วย 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();
}
}
}
คุณสามารถใช้คลาสสแกนเนอร์
Scanner sc=new Scanner(file);
sc.nextLine();
Scanner
ได้ดี แต่คำตอบนี้ไม่ได้รวมรหัสแบบเต็มเพื่อใช้งานอย่างถูกต้อง
BufferedReader.readLine()
จะเร็วกว่าหลายเท่า หากคุณคิดเป็นอย่างอื่นโปรดระบุเหตุผลของคุณ
คุณจำเป็นต้องใช้วิธีการในreadLine()
class BufferedReader
สร้างวัตถุใหม่จากคลาสนั้นและใช้วิธีการนี้กับเขาและบันทึกลงในสตริง
วิธีที่ชัดเจนในการบรรลุเป้าหมายนี้
ตัวอย่างเช่น:
หากคุณมี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();
}
}
try (Stream<String> stream = Files.lines(Paths.get(fileName))) {
stream.forEach(System.out::println);
}
System.getProperty("os.name").equals("Linux")
==
!
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);
ฉันมักจะทำกิจวัตรการอ่านตรงไปตรงมา:
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) {
}
}
}
คุณสามารถใช้รหัสนี้:
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();
}
}
}
โดยใช้แพ็คเกจorg.apache.commons.ioมันให้ประสิทธิภาพมากกว่าโดยเฉพาะในรหัสดั้งเดิมที่ใช้ Java 6 และด้านล่าง
Java 7 มี API ที่ดีกว่าโดยมีข้อยกเว้นน้อยกว่าในการจัดการและมีประโยชน์มากกว่า
LineIterator lineIterator = null;
try {
lineIterator = FileUtils.lineIterator(new File("/home/username/m.log"), "windows-1256"); // The second parameter is optionnal
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>
คุณยังสามารถใช้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();
}
FileUtils.readLines(file)
เป็นวิธีการเลิก นอกจากนี้เมธอดจะเรียกIOUtils.readLines
ใช้ซึ่งใช้ BufferedReader และ ArrayList นี่ไม่ใช่วิธีการต่อบรรทัดและไม่ใช่วิธีที่ใช้ได้จริงสำหรับการอ่านหลาย GB