Java.io.File
をbyte[]
に変換するにはどうすればいいですか?
それはあなたにとって最良の意味によります。生産性の面では、ホイールを再発明しないで、Apache Commonsを使用してください。これはここにあります IOUtils.toByteArray(InputStream input)
。
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());
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
JDK 7以降 - 1つのライナー
byte[] array = Files.readAllBytes(Paths.get("/path/to/file"));
外部の依存関係は必要ありません。
基本的にあなたはそれをメモリに読み込まなければなりません。ファイルを開き、配列を割り当て、ファイルの内容を配列に読み込みます。
最も簡単な方法はこれに似たものです。
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();
}
これはファイル内容の不必要なコピーをいくつか持っています(実際にはデータはファイルからbuffer
へ、buffer
からByteArrayOutputStream
へ、ByteArrayOutputStream
から実際の結果配列へ3回コピーされます)。
また、特定のサイズまでのメモリのみのファイルを読み込むようにする必要があります(これは通常アプリケーションに依存します):-)。
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
を参照)。
誰かが言ったように、 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);
...
}
}
あなたはそれを行うために同様に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を使用して以来、非常に高速だと思います。
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();
}
呼び出し側はストリームを閉じる責任があります。
// 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;
}
それをする簡単な方法:
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);
ファイルからバイトを読み込むための最も簡単な方法
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);
}
}
}
Guavaには Files.toByteArray() があります。いくつかの利点があります。
import Java.io.File;
import Java.nio.file.Files;
import Java.nio.file.Path;
File file = getYourFile();
Path path = file.toPath();
byte[] data = Files.readAllBytes(path);
コミュニティWikiの答えと同じアプローチを使いますが、箱から出してきれいにしてコンパイルします(たとえば、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();
}
これが最も簡単な方法だと思います。
org.Apache.commons.io.FileUtils.readFileToByteArray(file);
ReadFully このファイルからb.lengthバイトを現在のファイルポインタから始めてバイト配列に読み込みます。このメソッドは、要求されたバイト数が読み込まれるまでファイルから繰り返し読み込みます。このメソッドは、要求されたバイト数が読み取られるまで、ストリームの終わりが検出されるまで、または例外がスローされるまでブロックされます。
RandomAccessFile f = new RandomAccessFile(fileName, "r");
byte[] b = new byte[(int)f.length()];
f.readFully(b);
バイトを事前に割り当てられたバイトバッファに読み込む場合は、この回答が役立つ場合があります。
あなたの最初の推測はおそらく InputStream read(byte[])
を使うことでしょう。ただし、この方法には使用するのが不当に難しいという欠陥があります。たとえEOFが検出されなくても、配列が実際に完全に埋められるという保証はありません。
代わりに、 DataInputStream readFully(byte[])
を見てください。これは入力ストリームのラッパーであり、上記の問題はありません。さらに、このメソッドはEOFが見つかったときにスローされます。はるかに良いです。
他社製のライブラリを使用せずに別のソリューションを追加しましょう。 Scott ( link )によって提案された例外処理パターンを再利用します。そして私は醜い部分を別のメッセージに移動しました(私は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;
}
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);
}
}
}
ファイルからバイトを読み込む別の方法
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();
}
}
これを試して :
try
{
String path="";
InputStream inputStream=new FileInputStream(path);
byte[] data=IOUtils.readFully(inputStream,-1,false);
}
catch (Exception e)
{
}
「 import Sun.misc.IOUtils; 」を忘れないでください。
次の方法で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);
}
}
}