ฉันจะอ่านไฟล์ข้อความขนาดใหญ่ทีละบรรทัดโดยใช้ Java ได้อย่างไร


846

ฉันต้องอ่านไฟล์ข้อความขนาดใหญ่ประมาณ 5-6 GB ต่อบรรทัดโดยใช้ Java

ฉันจะทำสิ่งนี้ได้อย่างรวดเร็ว?


69
@kamaci และ อัล คำถามนี้ไม่ควรถูกทำเครื่องหมายว่าซ้ำซ้อน "อ่านบรรทัดสุดท้ายอย่างรวดเร็ว" ไม่ใช่ทางเลือกและเป็นที่ถกเถียงกันว่า "วิธีที่เร็วที่สุดในการอ่านบรรทัดไฟล์ข้อความโดยบรรทัด" คือ วิธีที่เร็วที่สุดในการทำบางสิ่งบางอย่างไม่จำเป็นต้องเป็นวิธีทั่วไป นอกจากนี้คำตอบด้านล่างนี้ยังรวมถึงรหัสซึ่งเป็นทางเลือกที่เกี่ยวข้องที่สุดที่คุณไม่มี คำถามนี้มีประโยชน์ ปัจจุบันเป็นผลการค้นหายอดนิยมของ google สำหรับ "java อ่านไฟล์ทีละบรรทัด" ในที่สุดการปิดเพื่อไปถึงสแต็คล้นและพบว่า 1 ในทุก ๆ 2 คำถามถูกตั้งค่าสถานะเพื่อการกำจัด
Patrick Cullen

5
นี่คือการเปรียบเทียบความเร็วสำหรับการใช้งานที่เป็นไปได้หกแบบ
Serg M Ten

4
เหตุการณ์แม้ว่าฉันได้อ่านความคิดเห็นที่พิสูจน์ว่านโยบายใกล้ชิดของ SO นั้นแย่มากดังนั้นก็ยังคงมีอยู่ มันเป็นมุมมองของนักพัฒนาที่มีใจที่แคบเพื่อหลีกเลี่ยงความซ้ำซ้อนในทุกค่าใช้จ่าย! แค่ปล่อยให้มันเป็น! ครีมจะขึ้นไปด้านบนและ sh * t จะจมลงไปด้านล่างได้ด้วยตัวเอง แม้ว่าอาจมีการถามคำถามก่อนหน้านี้ (ซึ่งคำถามนั้นไม่ใช่ ??) ซึ่งไม่ได้หมายความว่าคำถามใหม่อาจไม่สามารถวลีได้ดีขึ้นรับคำตอบที่ดีกว่าจัดอันดับสูงในเครื่องมือค้นหา ฯลฯ ที่น่าสนใจ ตอนนี้คำถามคือ 'ป้องกัน' ....
Stijn de Witt

3
มันเหลือเชื่อว่าคำถามจะถูกทำเครื่องหมายว่าซ้ำโดยเพียงแค่อ่านชื่อ
ลุค

คำตอบ:


1063

รูปแบบทั่วไปคือการใช้

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 จะทำในภายหลัง


6
รูปแบบนี้มีลักษณะอย่างไรกับการจัดการข้อยกเว้นที่เหมาะสม ฉันทราบว่า br.close () พ่น IOException ซึ่งดูน่าแปลกใจ - จะเกิดอะไรขึ้นเมื่อปิดไฟล์ที่เปิดอ่านแล้ว ตัวสร้างของ FileReader อาจส่งข้อยกเว้น FileNotFound
MikeB

3
หากฉันมีไฟล์ 200MB และสามารถอ่านได้ที่ 90MB / s ฉันคาดว่าจะใช้เวลาประมาณ 3 วินาที? ฉันดูเหมือนจะใช้เวลาไม่กี่นาทีด้วยวิธีการ "อ่านช้า" นี้ ฉันใช้ SSD ดังนั้นความเร็วในการอ่านไม่ควรเป็นปัญหา?
Jiew Meng

4
@ JiewMeng ดังนั้นฉันสงสัยว่าอย่างอื่นที่คุณกำลังทำคือการใช้เวลา คุณสามารถลองเพียงแค่อ่านบรรทัดของไฟล์และที่ไม่มีอะไรอื่น
Peter Lawrey

44
ทำไมไม่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 ) { ... } }ซึ่งยากที่จะไม่เกลียด
Aleksandr Dubinsky

26
@AleksandrDubinsky ปัญหาที่ฉันมีเกี่ยวกับการปิดใน Java 8 คือมันง่ายมากที่จะทำให้โค้ดอ่านง่ายขึ้น
Peter Lawrey

155

ดูบล็อกนี้:

อาจระบุขนาดบัฟเฟอร์หรืออาจใช้ขนาดเริ่มต้น ค่าเริ่มต้นมีขนาดใหญ่พอสำหรับวัตถุประสงค์ส่วนใหญ่

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

6
ไฟล์ของฉันคือ 1.5 Gig และมันเป็นไปไม่ได้ที่จะอ่านไฟล์โดยใช้คำตอบของคุณ!
Aboozar Rajabi

3
@AboozarRajabi แน่นอนมันเป็นไปได้ รหัสนี้สามารถอ่านไฟล์ข้อความใด ๆ
user207421

10
ลงคะแนนสำหรับลิงก์คุณภาพต่ำ ไม่มีจุดหมายอย่างสมบูรณ์DataInputStreamและกระแสที่ไม่ถูกต้องถูกปิด ไม่มีอะไรผิดปกติกับ Java Tutorial และไม่จำเป็นต้องอ้างถึงอินเทอร์เน็ตของบุคคลที่สามที่ไม่ชอบอะไรเช่นนี้
user207421

1
ฉันเขียนความคิดเห็นคุณมี 4 บรรทัด 100% ความคิดเห็นซ้ำซ้อนสำหรับรหัส 6 บรรทัด
บัฟฟาโล

97

เมื่อ 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);
}

1
ใช้StandardCharsets.UTF_8ใช้Stream<String>เพื่อความรัดกุมและหลีกเลี่ยงการใช้forEach()และโดยเฉพาะอย่างยิ่งforEachOrdered()ถ้าไม่มีเหตุผล
Aleksandr Dubinsky

2
ทำไมต้องหลีกเลี่ยง forEach () มันไม่ดีเหรอ?
steventrouble

ถ้าฉันเราแต่ละคนแทน forEachOrdered บรรทัดนั้นอาจพิมพ์ออกมาตามลำดับใช่ไหม?
msayag

2
@steventrouble ลองดูที่: stackoverflow.com/questions/16635398/...มันไม่ดีถ้าคุณผ่านการอ้างอิงฟังก์ชั่นสั้น ๆ เช่นแต่ได้รับน่าเกลียดถ้าคุณเขียนบล็อกของรหัสที่เป็นภายในforEach(this::process) lambdas forEach()
Aleksandr Dubinsky

2
@msayag คุณพูดถูกคุณจำเป็นต้องforEachOrderedดำเนินการตามคำสั่ง โปรดระวังว่าคุณจะไม่สามารถขนานกระแสข้อมูลในกรณีนั้นแม้ว่าฉันจะพบว่าการขนานไม่เปิดเว้นแต่ว่าไฟล์จะมีหลายพันบรรทัด
Aleksandr Dubinsky

38

นี่คือตัวอย่างที่มีการจัดการข้อผิดพลาดแบบเต็มและรองรับข้อมูลจำเพาะชุดอักขระสำหรับ 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;
    }
}

1
สิ่งที่ByteArrayInputStreamfed โดยสตริงที่แท้จริงเกี่ยวข้องกับการอ่านไฟล์ข้อความขนาดใหญ่?
user207421

ปิดไร้ประโยชน์อย่างแน่นอน ไม่มีเหตุผลที่จะปิดทุกสตรีม หากคุณปิดสตรีมเหล่านี้คุณจะปิดสตรีมอื่น ๆ โดยอัตโนมัติ ...
Enerccio

21

ใน 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>)) ในการทำงาน
เตฟาน

ฉันจะข้ามบรรทัดแรกด้วยวิธีนี้ได้อย่างไร
qed

2
@qedfor(String line : (Iterable<String>) lines.skip(1)::iterator)
Aleksandr Dubinsky

1
หากคุณไม่ได้ตั้งใจที่จะใช้Streamคุณสมบัติจริง ๆให้ใช้Files.newBufferedReaderแทนFiles.linesและโทรซ้ำ ๆreadLine()จนกระทั่งnullแทนที่จะใช้สิ่งก่อสร้าง(Iterable<String>) lines::iteratorดูเหมือนว่าจะง่ายกว่ามาก ...
Holger

ทำไมคุณถึงใช้ :: in lines :: iterator? การใช้งานที่ฉันรู้สำหรับ :: คือการตั้งชื่อวิธีการในฟังก์ชั่นแลมบ์ดา ในการวนรอบพารามิเตอร์หลังจาก: ควรเป็นตัวแปรในขณะที่คุณได้รับวิธี lambda โดยใช้ ::
Trismegistos

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

เครื่องสแกนโดยทั่วไปสแกนข้อความทั้งหมด ขณะที่ห่วงถูกนำมาใช้เพื่อสำรวจข้อความทั้งหมด

.hasNextLine()ฟังก์ชั่นบูลที่ส่งกลับจริงถ้ามีสายยังคงมากขึ้นในข้อความ .nextLine()ฟังก์ชั่นช่วยให้คุณทั้งเส้นเป็นสตริงซึ่งคุณสามารถใช้วิธีการที่คุณต้องการ ลองSystem.out.println(line)พิมพ์ข้อความ

หมายเหตุด้านข้าง: .txt เป็นข้อความประเภทไฟล์


ไม่ควรดูวิธีการประกาศแทน:´public คงเป็นโมฆะ readText พ่น FileNotFoundException () {´ชอบ: oid สาธารณะคงเป็นโมฆะ readText () พ่น FileNotFoundException {´
Ketcomp

ซึ่งช้ากว่าBufferedReader.readLine()มากและเขาขอวิธีที่ดีที่สุด
user207421

18

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) ดังนั้นคุณต้องระบุการเข้ารหัส


17

ฉันบันทึกและทดสอบ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 การอ่านไฟล์ที่นี่


1
ไกด์ของคุณน่าทึ่งมาก :)
Faisal Julaidan

คุณมักจะเป็นจังหวะSystem.out.print/println()ที่นี่ คุณกำลังสมมติว่าไฟล์จะพอดีกับหน่วยความจำในสองกรณีแรกของคุณ
user207421

ยุติธรรมพอสมควร บางทีฉันอาจทำให้สมมติฐานเหล่านั้นชัดเจนขึ้นในคำตอบของฉัน
gomisha

16

ใน 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, Marcelo
Marcelo Finki

CSV: ไฟล์คั่นด้วยเครื่องหมายจุลภาคดังนั้นคุณไม่ควรใช้เครื่องหมายจุลภาคในฟิลด์ csv ยกเว้นว่าคุณต้องการเพิ่มฟิลด์อื่น ดังนั้นใช้ split สำหรับเครื่องหมายจุลภาคในภาษาจาวาเมื่อทำการแยกไฟล์ CSV ออกมาได้อย่างสมบูรณ์แบบและถูกต้อง
Diego Duarte

7
ดิเอโกสิ่งนี้ไม่ถูกต้อง มาตรฐาน CSV เดียว (RFC 4180) กล่าวโดยเฉพาะว่า "ฟิลด์ที่มีตัวแบ่งบรรทัด (CRLF) เครื่องหมายคำพูดคู่และเครื่องหมายจุลภาคควรอยู่ในเครื่องหมายคำพูดคู่"
serg.nechaev

2
ใช้StandardCharsets.UTF_8เพื่อหลีกเลี่ยงข้อยกเว้นที่ตรวจสอบในCharset.forName("UTF-8")
Aleksandr Dubinsky

2
ขอบคุณ "Diego Duarte" สำหรับความคิดเห็นของคุณ ฉันต้องบอกว่าฉันเห็นด้วยกับสิ่งที่ "serg.nechaev" ตอบกลับ ฉันเห็นคอมม่าที่ฝังอยู่ในไฟล์ csv 'ตลอดเวลา' ผู้คนคาดหวังว่าสิ่งนี้จะได้รับการยอมรับ ด้วยความเคารพ ขอบคุณมากสำหรับ "serg.nechaev" IMHO คุณถูกต้อง เชิญทุกคน
Marcelo Finki

13

ใน Java 8 Files.lines()นอกจากนี้ยังมีทางเลือกที่จะใช้ หากแหล่งอินพุตของคุณไม่ใช่ไฟล์ แต่มีอะไรที่เป็นนามธรรมมากกว่าเช่นReaderหรือInputStreamคุณสามารถสตรีมบรรทัดผ่านเมธอดBufferedReaderslines()

ตัวอย่างเช่น:

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

จะเรียกสำหรับสายการป้อนข้อมูลแต่ละอ่านได้โดยprocessLine()BufferedReader


10

สำหรับการอ่านไฟล์ด้วย 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 sc=new Scanner(file);
sc.nextLine();

2
@Tim 'Bomb horribly' ไม่ใช่คำที่ฉันรู้จักใน CS คุณหมายถึงอะไรกันแน่?
user207421

กระโจนลงไปรันช้ามากผิดพลาดมากที่สุด ฉันอาจจะหลีกเลี่ยงสำนวนในเว็บไซต์นี้;)
ทิม

4
@Tim ทำไมมันถึงทำเช่นนั้น?
xehpuk

2
ใช้งานScannerได้ดี แต่คำตอบนี้ไม่ได้รวมรหัสแบบเต็มเพื่อใช้งานอย่างถูกต้อง
Aleksandr Dubinsky

5
@Tim รหัสนี้จะไม่ 'ระเบิดอย่างน่ากลัว' หรือ 'ชะงักลง' หรือ 'รันช้ามาก' หรือ 'ผิดพลาดได้มากที่สุด' ตามความเป็นจริงตามที่เขียนมันจะอ่านเพียงหนึ่งบรรทัดเกือบจะในทันที คุณสามารถอ่านเมกะไบต์ต่อวินาทีด้วยวิธีนี้แม้ว่าBufferedReader.readLine()จะเร็วกว่าหลายเท่า หากคุณคิดเป็นอย่างอื่นโปรดระบุเหตุผลของคุณ
user207421

7

คุณจำเป็นต้องใช้วิธีการในreadLine() class BufferedReaderสร้างวัตถุใหม่จากคลาสนั้นและใช้วิธีการนี้กับเขาและบันทึกลงในสตริง

BufferReader Javadoc


ดูเหมือนว่าลิงก์ไปยัง BufferReaderAPI เสีย
Sandeep

6

วิธีที่ชัดเจนในการบรรลุเป้าหมายนี้

ตัวอย่างเช่น:

หากคุณมี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();
    }
}

การส่งออกเช่นด้านล่าง ป้อนคำอธิบายรูปภาพที่นี่


ทำไมมันชัดเจน? และอย่าโพสต์ภาพข้อความที่นี่ โพสต์ข้อความ
user207421

คุณโพสต์รูปภาพ มันเป็นภาพของข้อความ คุณสามารถตัดและวางข้อความลงในหน้านี้โดยตรง ไม่มีใครพูดอะไรเกี่ยวกับการโพสต์โปรแกรม การโพสต์รูปภาพของข้อความเป็นการเสียเวลาของคุณซึ่งฉันไม่สนใจและแบนด์วิธของคุณไม่ดีเท่าที่ควร
user207421

6

Java 9:

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

2
ฉันคิดว่าคุณต้องSystem.getProperty("os.name").equals("Linux")
SpringLearner

5
อย่าเปรียบเทียบสตริงด้วย==!
JonasCz - Reinstate Monica

6
นี่คือตัวอย่างที่ยอมรับตามมาตรฐาน Java 8 ตามที่โพสต์แล้วโดยผู้อื่น ทำไมคุณถึงอ้างว่านี่คือ“ Java-9”?
Holger

@ โฮลเกอร์หน่วยความจำที่แมปไฟล์ที่เขาลืมพูดถึงอาจจะเป็นอย่างไร
ยูจีน

เพื่อดำเนินการทีละบรรทัดคุณสามารถลอง (สตรีม <String> สตรีม = Files.lines (Paths.get (inputFile))) {stream.forEach ((บรรทัด) -> {System.out.println (บรรทัด);} ); }
thanos.a

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

มันใช้งานได้สำหรับฉัน หวังว่ามันจะช่วยคุณได้เช่นกัน


3

คุณสามารถใช้สตรีมเพื่อทำอย่างแม่นยำมากขึ้น:

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

2
ฉันยอมรับว่ามันใช้ได้จริง ขั้นตอนที่คนไม่ชอบเพราะเลือก StringBuffer แปลก (StringBuilder เป็นที่ต้องการโดยทั่วไปถึงแม้ว่ามันอาจจะเป็นเพียงชื่อที่ไม่ดีสำหรับตัวแปร) อีกทั้งเพราะมันถูกกล่าวถึงแล้วข้างต้น
Andrii Rubtsov

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

คุณสามารถใช้รหัสนี้:

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

    }

}

คำอธิบายจะอยู่ในลำดับ
Peter Mortensen

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"); // The second parameter is optionnal
    while (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>

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 นี่ไม่ใช่วิธีการต่อบรรทัดและไม่ใช่วิธีที่ใช้ได้จริงสำหรับการอ่านหลาย GB
vallismortis
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.