web-dev-qa-db-ja.com

Javaを使用して大きなテキストファイルを1行ずつ読み取る方法

Javaを使用して、1行に5〜6 GB程度の大きなテキストファイルを読み込む必要があります。

どうすればこれを素早く実行できますか?

770
manoj singh

一般的なパターンは

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

注:#closeメソッドが確実に呼び出されるようにするには、Streamを try-with-resource ブロックに配置する必要があります。

953
Peter Lawrey

このブログを見てください。

バッファサイズを指定するか、デフォルトのサイズを使用することができます。デフォルトのは、ほとんどのの目的に十分な大きさです。

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

Java-8 が出たら(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);
}
82
msayag

これは、完全なエラー処理とJava 7より前のバージョンの文字セット指定のサポートを含むサンプルです。Java7では、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;
    }
}
35
DarkStar

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は、StreamをIterableにキャストします。

21

あなたができることは、Scannerを使用してテキスト全体をスキャンし、テキストを1行ずつ移動することです。

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()関数はあなたがそれからあなたが望む方法を使うことができるStringとしてあなたにあなたの全体の行を与えます。テキストを印刷するためにSystem.out.println(line)を試してください。

注意:.txtはファイルタイプのテキストです。

19
iskandarchacra

FileReaderではエンコーディングを指定できません。指定する必要がある場合はInputStreamReaderinsteadを使用してください。

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
live-love

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);
}
15
Diego Duarte

Javaでファイルを読むための10の異なる方法 を文書化してテストし、それからそれらを1KBから1GBのテストファイルで読むようにしてお互いに対抗しました。これは、1GBのテストファイルを読み取るための最速の3つのファイル読み取り方法です。

パフォーマンステストを実行するとき、コンソールには何も出力しませんでした。テストが遅くなるためです。私は生の読書速度をテストしたいだけでした。

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では機能しません。 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

Java 7、8、9で動作することがテストされています。これは、1GBのテストファイルを読み込むのに約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のファイル読み取り方法 ここ の完全なランキングを見つけることができます。

11
gomisha

読み取りファイル 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();
        }
        }
    }
10
Ankit Sood

Scannerクラスが使えます

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

Java 8では、 Files.lines() を使用する代わりの方法もあります。あなたの入力ソースがファイルではなくReaderInputStreamのようなもっと抽象的なものであるなら、BufferedReaders lines()メソッドを使って stream 行を書くことができます。

例えば:

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

BufferedReaderによって読み込まれた各入力行に対してprocessLine()を呼び出します。

8

あなたはclass BufferedReaderreadLine()メソッドを使う必要があります。

BufferReader Javadoc

7
Master C

Java-9:

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

これを達成するための明確な方法

例えば:

現在のディレクトリに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();
    }
}

以下のような出力、 enter image description here

5
Rajamohan S
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
Dipendra Ghatal

私は通常、読むことを簡単にします。 

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) {
        }
    }
}
2

org.Apache.commons.io パッケージを使用すると、特にJava 6以下を使用するレガシーコードでより高いパフォーマンスが得られました。
Java7は、より少ない例外処理とより有用なメソッドを持つより良い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>
1
mohsen.nour

ストリームを使ってより正確に行うことができます。

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

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

このコードを使うことができます:

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
Usman Yaqoob