Javaでbyte []するファイル


757

どのようにして変換しますjava.io.Filebyte[]


考えられる用途の1つは、ファイルからシリアル化されたオブジェクトを読み取ることです。
Mahm00d 14

2
もう1つは、ヘッダーを使用してファイルの種類を見つけることです。
James P.

このバイトを試す[]バイト= null; BufferedInputStream fileInputStream = null; try {File file = new File(filePath); fileInputStream = new BufferedInputStream(new FileInputStream(file)); // fileInputStream = Thread.currentThread()。getContextClassLoader()。getResourceAsStream(this.filePath); bytes = new byte [(int)file.length()]; fileInputStream.read(bytes); } catch(FileNotFoundException ex){スローex; }
Rohit Chaurasiya

回答:


486

それはあなたにとって最良の意味に依存します。生産性については、車輪を再発明してApache Commonsを使用しないでください。ここIOUtils.toByteArray(InputStream input)ですね。


29
@ymajoros:そうだね!さらに別の依存関係よりも、コード行を追加したほうがよいでしょう。依存関係には隠れたコストがあります。そのライブラリを最新の状態に保ち、ビルドスクリプトなどに依存関係を含め、コードを使用している人にそれを伝えるなどする必要があります。それを使用するよりもコードのあるライブラリを既に使用している場合は、他の人が言うでしょう自分で書いてください。
Stijn de Witt 2013

11
これは、ファイルの読み取り方法の質問には答えますが、タイプjava.IO.Fileのオブジェクトをbyte []に​​変換する方法の質問には答えません。
2013年

5
これはどのようにa Fileを読むのに使用されますbyte[]か?Java6を使用しているため、NIOメソッドを使用できません:(
PASTRY

4
@ymajorosは、「標準の3行ソリューション」を私たちと親切に共有してくれるので、車輪に依存する再発明に依存する必要はありませんか?
matteo 2014

3
@matteo:ありますか?他の回答、たとえばFiles.readAllBytes()を参照してください。シンプルで依存関係はありません。
ymajoros 14

1292

JDK 7から使用できますFiles.readAllBytes(Path)

例:

import java.io.File;
import java.nio.file.Files;

File file;
// ...(file is initialised)...
byte[] fileContent = Files.readAllBytes(file.toPath());

10
パスではなくFileオブジェクトがあります(http postリクエストから)
aldo.roman.nurena '28

81
@ aldo.roman.nurena JDK7は、パスオブジェクトを提供するFile.toPath()メソッドを導入しました。
KevinL 2013年

6
ファイルからパスを取得できます。試してください:ファイルfile = new File( "/ path"); パスパス= Paths.get(file.getAbsolutePath()); byte [] data = Files.readAllBytes(path);
gfelisberto 2013

2
ファイルのクローズはjava.nioでどのように処理されますか?つまり、上記のコードは何かをクローズする必要がありますか?
akauppi 2014年

4
@akauppi回答のリンクを参照してください:「このメソッドはファイルが閉じられることを保証します...」
Bernhard Barker

226

JDK 7以降-1つのライナー:

byte[] array = Files.readAllBytes(Paths.get("/path/to/file"));

外部依存関係は必要ありません。


13
これは、Apache Commonsを必要とする承認済みの回答よりも優れた選択肢です。
james.garriss

1
ありがとう:)私もこれを必要としました:String text = new String(Files.readAllBytes(new File( "/ path / to / file")。toPath())); これは元々、stackoverflow.com
a / 26888713/1257959

5
アンドロイドでは、それは26分であることをAPIレベルを必要とする
アッシュートッシュChamoli

2
追加する必要があります。まだ追加import java.nio.file.Files;import java.nio.file.Paths;ていない場合は追加してください。
サム

164
import java.io.RandomAccessFile;
RandomAccessFile f = new RandomAccessFile(fileName, "r");
byte[] b = new byte[(int)f.length()];
f.readFully(b);

Java 8のドキュメント:http : //docs.oracle.com/javase/8/docs/api/java/io/RandomAccessFile.html


2
f.read()の戻り値を確認する必要があります。ここで時々、ファイル全体を読み取れないことがあります。
bugs_ 2012

8
このような状況は、ファイルの読み取り中にファイルが変更されている場合にのみ発生します。その他の場合はすべて、IOExceptionがスローされます。この問題に対処するには、ファイルを読み取り/書き込みモードで開くことをお勧めします
。RandomAccessFile

5
ファイルの一部のみを読み取るための他のソース(ファイルはネットワーク共有上にある...)を想像できます。readFully()には、検索しているコントラクトがあります。
DThought

3
RandomAccessFileはスレッドセーフではないことに注意してください。したがって、場合によっては同期が必要になることがあります。
バンサー2013年

@DmitryMitskevich他のケースとして、非コンフォームの可能性のあるファイルシステムもあります。たとえば、Linuxの/ proc /で「ファイル」を読み取ると、読み取りが短くなる可能性があります(つまり、すべてを読み取るにはループが必要です)
nos

78

基本的には、メモリで読み取る必要があります。ファイルを開き、配列を割り当て、ファイルの内容を配列に読み込みます。

最も簡単な方法はこれに似たものです:

public byte[] read(File file) throws IOException, FileTooBigException {
    if (file.length() > MAX_FILE_SIZE) {
        throw new FileTooBigException(file);
    }
    ByteArrayOutputStream ous = null;
    InputStream ios = null;
    try {
        byte[] buffer = new byte[4096];
        ous = new ByteArrayOutputStream();
        ios = new FileInputStream(file);
        int read = 0;
        while ((read = ios.read(buffer)) != -1) {
            ous.write(buffer, 0, read);
        }
    }finally {
        try {
            if (ous != null)
                ous.close();
        } catch (IOException e) {
        }

        try {
            if (ios != null)
                ios.close();
        } catch (IOException e) {
        }
    }
    return ous.toByteArray();
}

これには、ファイルの内容の不必要なコピーがいくつかあります(実際には、データが3回コピーされます:ファイルからbufferbufferByteArrayOutputStream、からByteArrayOutputStream、実際の結果の配列へ)。

また、特定のサイズまでのメモリのみのファイルを読み取ることを確認する必要もあります(これは通常、アプリケーションに依存します):-)。

またIOException、関数の外側を扱う必要があります。

別の方法はこれです:

public byte[] read(File file) throws IOException, FileTooBigException {
    if (file.length() > MAX_FILE_SIZE) {
        throw new FileTooBigException(file);
    }

    byte[] buffer = new byte[(int) file.length()];
    InputStream ios = null;
    try {
        ios = new FileInputStream(file);
        if (ios.read(buffer) == -1) {
            throw new IOException(
                    "EOF reached while trying to read the whole file");
        }
    } finally {
        try {
            if (ios != null)
                ios.close();
        } catch (IOException e) {
        }
    }
    return buffer;
}

これには不要なコピーはありません。

FileTooBigExceptionカスタムアプリケーションの例外です。MAX_FILE_SIZE定数は、アプリケーションパラメータです。

大きなファイルの場合は、ストリーム処理アルゴリズムを考えるか、メモリマッピングを使用する必要があります(を参照java.nio)。


iosはトライの外で宣言する必要があります
ダリルスピッツァー

2番目の例の "ios.read(buffer)"ステートメントは、ファイルの最初の4096バイトのみを読み取ります(最初の例と同じ4kバッファーを想定)。2番目の例が機能するためには、結果が-1(ファイルの終わりに達した)かどうかを確認するwhileループ内にある必要があると思います。
Stijn de Witt 2013

申し訳ありませんが、上記の私の発言を却下し、ステートメントの長さをファイルの長さに設定するステートメントを逃しました。それでも、最初の例の方が好きです。ファイル全体を一度にバッファに読み込むことはスケーラブルではありません。ファイルが大きい場合は、メモリ不足になる可能性があります。
Stijn de Witt 2013

「最も簡単な」方法は、try-with-resourcesを利用することです。
Sina Madani 2016年

クールですが、少し冗長です。
Sapphire_Brick

77

誰かが言ったように、Apache Commons File Utilsはあなたが探しているものを持っているかもしれません

public static byte[] readFileToByteArray(File file) throws IOException

使用例(Program.java):

import org.apache.commons.io.FileUtils;
public class Program {
    public static void main(String[] args) throws IOException {
        File file = new File(args[0]);  // assume args[0] is the path to file
        byte[] data = FileUtils.readFileToByteArray(file);
        ...
    }
}

23

NIO APIを使用してそれを行うこともできます。合計ファイルサイズ(バイト単位)がintに収まる限り、このコードでこれを行うことができます。

File f = new File("c:\\wscp.script");
FileInputStream fin = null;
FileChannel ch = null;
try {
    fin = new FileInputStream(f);
    ch = fin.getChannel();
    int size = (int) ch.size();
    MappedByteBuffer buf = ch.map(MapMode.READ_ONLY, 0, size);
    byte[] bytes = new byte[size];
    buf.get(bytes);

} catch (IOException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
} finally {
    try {
        if (fin != null) {
            fin.close();
        }
        if (ch != null) {
            ch.close();
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

MappedByteBufferを使用しているため、非常に高速だと思います。


2
ファイルを1回だけ読み取る場合は、メモリマッピングを使用する必要はまったくなく、通常のFileInputStreamを使用する場合の2倍のメモリを使用します。
ジェームズ

1
残念ながら、MappedByteBufferは自動的に解放されません。
トム・ホーティン-タックライン09年

2
すばらしい、新しい例には、古典的な壊れた例外処理であるprintStackTraceが含まれています。
ジェームズ、

私は同意します。Eclipseがデフォルトで使用するものです。例外を再スローする必要があると思います。
アミット

ファイルからbyte []を作成するために、nioのベンチマークを行っています。直接バッファを使用する以外は、実際には2倍のメモリが必要です。非常に大きなファイルの場合は高速ですが(200Mの場合のバッファー付きIOの約2倍)、約5Mのファイルの場合は5倍失われるようです。
2014年

22

Java 8がなく、数行のコードの記述を避けるために大規模なライブラリを含めることは悪い考えであると私に同意する場合:

public static byte[] readBytes(InputStream inputStream) throws IOException {
    byte[] b = new byte[1024];
    ByteArrayOutputStream os = new ByteArrayOutputStream();
    int c;
    while ((c = inputStream.read(b)) != -1) {
        os.write(b, 0, c);
    }
    return os.toByteArray();
}

呼び出し元は、ストリームを閉じる責任があります。


21
// Returns the contents of the file in a byte array.
    public static byte[] getBytesFromFile(File file) throws IOException {        
        // Get the size of the file
        long length = file.length();

        // You cannot create an array using a long type.
        // It needs to be an int type.
        // Before converting to an int type, check
        // to ensure that file is not larger than Integer.MAX_VALUE.
        if (length > Integer.MAX_VALUE) {
            // File is too large
            throw new IOException("File is too large!");
        }

        // Create the byte array to hold the data
        byte[] bytes = new byte[(int)length];

        // Read in the bytes
        int offset = 0;
        int numRead = 0;

        InputStream is = new FileInputStream(file);
        try {
            while (offset < bytes.length
                   && (numRead=is.read(bytes, offset, bytes.length-offset)) >= 0) {
                offset += numRead;
            }
        } finally {
            is.close();
        }

        // Ensure all the bytes have been read in
        if (offset < bytes.length) {
            throw new IOException("Could not completely read file "+file.getName());
        }
        return bytes;
    }

また、numReadをループ内に配置します。できる限り最小の有効スコープで変数を宣言します。whileループの外側に置くことは、その複雑な「while」テストを有効にするためにのみ必要です。ループ内でEOFのテストを行うことをお勧めします(発生した場合はEOFExceptionをスローします)。
エリクソン2009年

throw new IOException("File is too large!");ファイルが大きすぎる場合はどうすればよいですか?それについて何か例はありますか?
2015

21

それを行う簡単な方法:

File fff = new File("/path/to/file");
FileInputStream fileInputStream = new FileInputStream(fff);

// int byteLength = fff.length(); 

// In android the result of file.length() is long
long byteLength = fff.length(); // byte count of the file-content

byte[] filecontent = new byte[(int) byteLength];
fileInputStream.read(filecontent, 0, (int) byteLength);

すでに述べたワンライナーなど、もっと簡単な方法があります。
Sapphire_Brick

@Sapphire_Brickより簡単な方法ですが、ワンライナーはすべての状況に適合するわけではありません。Androidなど。
Behr

17

ファイルからバイトを読み取る最も簡単な方法

import java.io.*;

class ReadBytesFromFile {
    public static void main(String args[]) throws Exception {
        // getBytes from anyWhere
        // I'm getting byte array from File
        File file = null;
        FileInputStream fileStream = new FileInputStream(file = new File("ByteArrayInputStreamClass.java"));

        // Instantiate array
        byte[] arr = new byte[(int) file.length()];

        // read All bytes of File stream
        fileStream.read(arr, 0, arr.length);

        for (int X : arr) {
            System.out.print((char) X);
        }
    }
}

1
私は「
最も簡単な

ここで説明できますか?なぜあなたは議論をしているのですか?
ムハンマドサディク2017

3
特別なことは何もありませんが、あなたは最も単純であると言い、私はより単純な解決策を見つけます->私の意見では、それは最も単純ではありません。おそらく数年前だったかもしれませんが、世界は変化しています。私は自分の解決策にそのような発言を付けません。;)あなただけが「私の意見では、最も単純なのは..」または「私が見つけた最も単純なのは...」と書いた場合、気になりたくないので、これを伝えて良かったと思ってください。
BlondCode 2017

@MuhammadSadiq:何もインポートしないでください。これは.*悪い習慣と見なされています。
Sapphire_Brick

13

Guavaには、提供するFiles.toByteArray()があります。これにはいくつかの利点があります。

  1. これは、ファイルの長さが0と報告されていてもコンテンツが含まれているコーナーケースをカバーしています
  2. これは高度に最適化されており、大きなファイルを読み込もうとすると、ファイルを読み込もうとするとOutOfMemoryExceptionが発生します。(file.length()の巧妙な使用を通じて)
  3. ホイールを再発明する必要はありません。


11

コミュニティウィキの回答と同じアプローチを使用しますが、よりクリーンですぐにコンパイルできます(AndroidなどでApache Commonsライブラリをインポートしたくない場合は、推奨されるアプローチ)。

public static byte[] getFileBytes(File file) throws IOException {
    ByteArrayOutputStream ous = null;
    InputStream ios = null;
    try {
        byte[] buffer = new byte[4096];
        ous = new ByteArrayOutputStream();
        ios = new FileInputStream(file);
        int read = 0;
        while ((read = ios.read(buffer)) != -1)
            ous.write(buffer, 0, read);
    } finally {
        try {
            if (ous != null)
                ous.close();
        } catch (IOException e) {
            // swallow, since not that important
        }
        try {
            if (ios != null)
                ios.close();
        } catch (IOException e) {
            // swallow, since not that important
        }
    }
    return ous.toByteArray();
}


7

ReadFullyこのファイルからバイト配列にb.lengthバイトを読み取ります。現在のファイルポインターから開始します。このメソッドは、要求されたバイト数が読み取られるまで、ファイルから繰り返し読み取ります。このメソッドは、要求されたバイト数が読み取られるか、ストリームの終わりが検出されるか、または例外がスローされるまでブロックします。

RandomAccessFile f = new RandomAccessFile(fileName, "r");
byte[] b = new byte[(int)f.length()];
f.readFully(b);

5

事前に割り当てられたバイトバッファーにバイトを読み取る場合は、この回答が役立ちます。

あなたの最初の推測はおそらく使用することでしょうInputStream read(byte[])。ただし、この方法には、使用が不当に困難になる欠点があります。EOFに遭遇しなくても、配列が実際に完全に満たされる保証はありません。

代わりに、をご覧くださいDataInputStream readFully(byte[])。これは入力ストリームのラッパーであり、上記の問題はありません。さらに、このメソッドは、EOFが検出されるとスローされます。ずっといい。


4

次の方法はjava.io.Fileをbyte []に​​変換するだけでなく、多くの異なるJavaファイル読み取りメソッドをテストするときに、ファイルを読み取る最も速い方法であることがわかりましたを相互。

java.nio.file.Files.readAllBytes()

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-10KB.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);
    }
  }
}

3

サードパーティのライブラリを使用せずに別のソリューションを追加しましょう。Scottリンク)によって提案された例外処理パターンを再利用します。そして、醜い部分を別のメッセージに移動しました(FileUtilsクラスで非表示にします;))

public void someMethod() {
    final byte[] buffer = read(new File("test.txt"));
}

private byte[] read(final File file) {
    if (file.isDirectory())
        throw new RuntimeException("Unsupported operation, file "
                + file.getAbsolutePath() + " is a directory");
    if (file.length() > Integer.MAX_VALUE)
        throw new RuntimeException("Unsupported operation, file "
                + file.getAbsolutePath() + " is too big");

    Throwable pending = null;
    FileInputStream in = null;
    final byte buffer[] = new byte[(int) file.length()];
    try {
        in = new FileInputStream(file);
        in.read(buffer);
    } catch (Exception e) {
        pending = new RuntimeException("Exception occured on reading file "
                + file.getAbsolutePath(), e);
    } finally {
        if (in != null) {
            try {
                in.close();
            } catch (Exception e) {
                if (pending == null) {
                    pending = new RuntimeException(
                        "Exception occured on closing file" 
                             + file.getAbsolutePath(), e);
                }
            }
        }
        if (pending != null) {
            throw new RuntimeException(pending);
        }
    }
    return buffer;
}

3
public static byte[] readBytes(InputStream inputStream) throws IOException {
    byte[] buffer = new byte[32 * 1024];
    int bufferSize = 0;
    for (;;) {
        int read = inputStream.read(buffer, bufferSize, buffer.length - bufferSize);
        if (read == -1) {
            return Arrays.copyOf(buffer, bufferSize);
        }
        bufferSize += read;
        if (bufferSize == buffer.length) {
            buffer = Arrays.copyOf(buffer, bufferSize * 2);
        }
    }
}

1

ファイルからバイトを読み取る別の方法

Reader reader = null;
    try {
        reader = new FileReader(file);
        char buf[] = new char[8192];
        int len;
        StringBuilder s = new StringBuilder();
        while ((len = reader.read(buf)) >= 0) {
            s.append(buf, 0, len);
            byte[] byteArray = s.toString().getBytes();
        }
    } catch(FileNotFoundException ex) {
    } catch(IOException e) {
    }
    finally {
        if (reader != null) {
            reader.close();
        }
    }

中空のキャッチブロックは使用しないでください。デバッグが難しくなります。
Sapphire_Brick

1
//The file that you wanna convert into byte[]
File file=new File("/storage/0CE2-EA3D/DCIM/Camera/VID_20190822_205931.mp4"); 

FileInputStream fileInputStream=new FileInputStream(file);
byte[] data=new byte[(int) file.length()];
BufferedInputStream bufferedInputStream=new BufferedInputStream(fileInputStream);
bufferedInputStream.read(data,0,data.length);

//Now the bytes of the file are contain in the "byte[] data"

1
このコードは質問に対する解決策を提供するかもしれませんが、それがなぜ/どのように機能するかに関してコンテキストを追加する方が良いです。これは、将来のユーザーが学び、その知識を自分のコードに適用するのに役立ちます。また、コードが説明されている場合は、賛成票の形でユーザーから肯定的なフィードバックを受ける可能性があります。
borchvm

それは、今後の投稿で覚えておきたい重要な部分です。有益な洞察をありがとう。
ウサマメフムード

0

これを試して :

import sun.misc.IOUtils;
import java.io.IOException;

try {
    String path="";
    InputStream inputStream=new FileInputStream(path);
    byte[] data=IOUtils.readFully(inputStream,-1,false);
}
catch (IOException e) {
    System.out.println(e);
}

それには、別のJREで実行するとアプリが機能しなくなる特定のJRE実装が必要です。
ラタマン

2
小さな間違い:IOExceptionであり、IOexceptionではありませんが、感謝します:)
Marciano

1
@MatanMarciano:私の悪い点
Sapphire_Brick

-7

ではJDK8

Stream<String> lines = Files.lines(path);
String data = lines.collect(Collectors.joining("\n"));
lines.close();

2
フランス語を話す私の友人の質問を読んでください、それは「byte []」への変換について尋ねます、そしてあなたの答えはそれを提供しません。
カイザーキースター

2
これは、byte []に​​変換するために応答するリモートオプションを提供しません。
Anddo
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.