Это зависит от того, что лучше для вас значит. Производительность мудрая, не изобретайте велосипед и используйте Apache Commons. Который здесь IOUtils.toByteArray(InputStream input).
@ymajoros: так правда! Я предпочел бы иметь несколько дополнительных строк кода, чем еще одну зависимость. Зависимости имеют скрытые затраты. Вам нужно быть в курсе этой библиотеки, включать зависимости в свои скрипты сборки и т. Д., Сообщать об этом людям, использующим ваш код и т. Д. И т. Д. Если вы уже используете библиотеку, в которой есть код, чем ее используйте, в противном случае, я бы сказал, напиши это сам.
Стейн де Витт
11
Это отвечает на вопрос о том, как читать файл, но не на вопрос о том, как преобразовать объект типа java.IO.File в byte [].
Инго
5
Как это используется , чтобы читать , Fileчтобы byte[]? Я использую Java6, поэтому я не могу использовать методы NIO :(
PASTRY
4
@ymajoros, не могли бы вы поделиться с нами какими-либо «стандартными 3-строчными решениями», чтобы нам не приходилось полагаться на переосмысление зависимости?
Маттео
3
@matteo: любой? Смотрите другие ответы, например Files.readAllBytes (). Просто, без зависимости.
У меня есть объект File, а не путь (из запроса HTTP http)
aldo.roman.nurena
82
@ aldo.roman.nurena JDK7 представил метод File.toPath (), который даст вам объект Path.
KevinL
7
Вы можете получить путь из файла. Попробуйте: Файл файл = новый файл ("/ путь"); Path path = Paths.get (file.getAbsolutePath ()); byte [] data = Files.readAllBytes (path);
gfelisberto
2
Как происходит закрытие файла в java.nio - другими словами, должен ли приведенный выше код что-то закрывать?
akauppi
4
@akauppi См. ссылку в ответе: «Метод обеспечивает закрытие файла ...»
Теперь это лучший выбор, чем принятый ответ, который требует Apache Commons.
james.garriss
1
Спасибо :) Мне также был нужен этот: String text = new String (Files.readAllBytes (new File ("/ path / to / file"). ToPath ())); который первоначально от stackoverflow.com/a/26888713/1257959
cgl
5
В Android требуется минимальный уровень API, чтобы быть 26.
Ашутош Чамоли
2
Вам нужно будет добавить, import java.nio.file.Files;и import java.nio.file.Paths;если вы еще не сделали.
Сэм
164
import java.io.RandomAccessFile;RandomAccessFile f =newRandomAccessFile(fileName,"r");byte[] b =newbyte[(int)f.length()];
f.readFully(b);
Вы должны проверить возвращаемое значение f.read (). Иногда может случиться так, что вы не будете читать весь файл.
bugs_
8
Такая ситуация может возникнуть, только если файл меняется во время чтения. Во всех остальных случаях IOException выбрасывается. Для решения этой проблемы предлагаю открыть файл в режиме чтения-записи: RandomAccessFile (fileName, "rw")
Дмитрий Мицкевич
5
Я мог бы представить другие источники только для чтения части файла (Файл находится в общей сетевой папке ...) readFully () имеет контракт, который вы ищете.
Думал
3
Помните, что RandomAccessFile не является потокобезопасным. Поэтому в некоторых случаях может потребоваться синхронизация.
bancer
@DmitryMitskevich Есть и другие случаи на файловых системах, которые, возможно, не соответствуют. например, чтение «файлов» в / proc / on linux может вызвать короткие чтения (т. е. вам нужен цикл, чтобы прочитать все это)
nos
78
В основном вы должны прочитать это в памяти. Откройте файл, выделите массив и прочитайте содержимое файла в массив.
Это приводит к некоторому ненужному копированию содержимого файла (фактически данные копируются три раза: из файла в buffer, из, bufferв ByteArrayOutputStream, ByteArrayOutputStreamв реальный результирующий массив).
Вам также необходимо убедиться, что вы читаете в памяти только файлы определенного размера (обычно это зависит от приложения) :-).
Вы также должны относиться к IOExceptionвнешней функции.
Другой способ заключается в следующем:
publicbyte[] read(File file)throwsIOException,FileTooBigException{if(file.length()> MAX_FILE_SIZE){thrownewFileTooBigException(file);}byte[] buffer =newbyte[(int) file.length()];InputStream ios =null;try{
ios =newFileInputStream(file);if(ios.read(buffer)==-1){thrownewIOException("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Константа AN параметры приложения.
Для больших файлов вы, вероятно, должны подумать об алгоритме обработки потока или использовать отображение памяти (см. java.nio).
Оператор «ios.read (buffer)» во втором примере будет читать только первые 4096 байт файла (при условии, что используется тот же буфер 4 КБ, который использовался в первом примере). Чтобы второй пример работал, я думаю, что чтение должно быть внутри цикла while, который проверяет результат на -1 (достигнут конец файла).
Стейн де Витт
Извините, отклонил мое замечание выше, пропустил буфер настроек оператора по длине файла. Тем не менее, мне больше нравится первый пример. Чтение всего файла в буфер за один раз не масштабируется. Вы рискуете загорать из памяти, когда файл большой.
Стейн де Витт
«Самый простой» способ - использовать попытку с ресурсами.
import org.apache.commons.io.FileUtils;publicclassProgram{publicstaticvoid main(String[] args)throwsIOException{File file =newFile(args[0]);// assume args[0] is the path to filebyte[] data =FileUtils.readFileToByteArray(file);...}}
абсолютно не нужно использовать отображение памяти, если вы собираетесь прочитать файл только один раз, и в итоге он будет использовать вдвое больше памяти, чем обычный FileInputStream.
Джеймс
1
К сожалению, MappedByteBuffer не выпускается автоматически.
Том Хотин - tackline
2
офигенно, новый пример включает printStackTrace, классическую обработку неработающих исключений.
Джеймс
Я согласен .. Это затмение по умолчанию, которое вставляет затмение. Я думаю, что я должен отбросить исключение!
Амит
Я тестировал nio, чтобы создать байт [] из файла. Помимо использования прямого буфера, он действительно занимает вдвое больше памяти. Хотя он работает быстрее для очень больших файлов (примерно в два раза быстрее, чем буферизованный ввод-вывод для 200M), он проигрывает в 5 раз для файлов около 5M.
Chaffers
22
Если у вас нет Java 8, и вы согласны со мной, что включение массивной библиотеки во избежание написания нескольких строк кода - плохая идея:
publicstaticbyte[] readBytes(InputStream inputStream)throwsIOException{byte[] b =newbyte[1024];ByteArrayOutputStream os =newByteArrayOutputStream();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.publicstaticbyte[] getBytesFromFile(File file)throwsIOException{// Get the size of the filelong 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 largethrownewIOException("File is too large!");}// Create the byte array to hold the databyte[] bytes =newbyte[(int)length];// Read in the bytesint offset =0;int numRead =0;InputStream is =newFileInputStream(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 inif(offset < bytes.length){thrownewIOException("Could not completely read file "+file.getName());}return bytes;}
Кроме того, поместите numRead внутри цикла. Объявите переменные в наименьшей допустимой области видимости. Помещение его вне цикла while необходимо только для включения этого сложного теста «while»; было бы лучше сделать тест на EOF внутри цикла (и выбросить EOFException, если это произойдет).
Эриксон
throw new IOException("File is too large!");что делать, если файл слишком большой? Есть ли еще пример?
Фер
21
Простой способ сделать это:
File fff =newFile("/path/to/file");FileInputStream fileInputStream =newFileInputStream(fff);// int byteLength = fff.length(); // In android the result of file.length() is longlong byteLength = fff.length();// byte count of the file-contentbyte[] filecontent =newbyte[(int) byteLength];
fileInputStream.read(filecontent,0,(int) byteLength);
Я спорю о том, чтобы быть «самым простым путем» :)
BlondCode
Вы можете объяснить здесь? Почему вы спорите?
Мухаммед Садик
3
Ничего особенного, но вы говорите самое простое, и я вижу более простые решения -> на мой взгляд, это не самое простое. Может быть, это было пару лет назад, но мир меняется. Я бы не назвал свои собственные решения таким заявлением. ;) Если бы вы только написали: «На мой взгляд, самое простое - это ...» или «самое простое, что я нашел…». Не хочу вас беспокоить, просто подумайте, как приятно это донести.
BlondCode
@MuhammadSadiq: ничего не импортируйте .*, это считается плохой практикой.
Sapphire_Brick
13
У Guava есть Files.toByteArray (), чтобы предложить вам. У этого есть несколько преимуществ:
Это покрывает угловой случай, когда файлы сообщают о длине 0, но все еще имеют содержание
Он очень оптимизирован, вы получаете исключение OutOfMemoryException, если пытаетесь прочитать большой файл, даже не пытаясь загрузить файл. (Через умное использование file.length ())
Используя тот же подход, что и вики-ответ сообщества, но более чистый и компилируемый из коробки (предпочтительный подход, если вы не хотите импортировать библиотеки Apache Commons, например, на Android):
publicstaticbyte[] getFileBytes(File file)throwsIOException{ByteArrayOutputStream ous =null;InputStream ios =null;try{byte[] buffer =newbyte[4096];
ous =newByteArrayOutputStream();
ios =newFileInputStream(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();}
уже есть ответ с этим предложением от Тома в 2009 году
Кнут Херрманн
7
ReadFully Считывает байты b.length из этого файла в байтовый массив, начиная с текущего указателя файла. Этот метод читает несколько раз из файла, пока не будет прочитано запрошенное количество байтов. Этот метод блокируется до тех пор, пока не будет прочитано запрошенное количество байтов, не будет обнаружен конец потока или не сгенерировано исключение.
RandomAccessFile f =newRandomAccessFile(fileName,"r");byte[] b =newbyte[(int)f.length()];
f.readFully(b);
Если вы хотите прочитать байты в заранее выделенный байтовый буфер, этот ответ может помочь.
Ваше первое предположение, вероятно, будет использовать InputStream read(byte[]). Однако этот метод имеет недостаток, который делает его неоправданно сложным: нет никакой гарантии, что массив будет фактически полностью заполнен, даже если EOF не обнаружен.
Вместо этого взгляните на DataInputStream readFully(byte[]). Это оболочка для входных потоков, которая не имеет вышеуказанной проблемы. Кроме того, этот метод выбрасывает при обнаружении EOF. Гораздо приятнее.
Мало того, что следующий способ преобразует файл java.io.File в байт [], я также обнаружил, что это самый быстрый способ чтения в файле при тестировании множества различных методов чтения Java-файлов. друг против друга:
Позвольте мне добавить другое решение без использования сторонних библиотек. Он повторно использует шаблон обработки исключений, который был предложен Скоттом ( ссылка ). И я переместил некрасивую часть в отдельное сообщение (я бы спрятался в каком-то классе FileUtils;))
publicvoid someMethod(){finalbyte[] buffer = read(newFile("test.txt"));}privatebyte[] read(finalFile file){if(file.isDirectory())thrownewRuntimeException("Unsupported operation, file "+ file.getAbsolutePath()+" is a directory");if(file.length()>Integer.MAX_VALUE)thrownewRuntimeException("Unsupported operation, file "+ file.getAbsolutePath()+" is too big");Throwable pending =null;FileInputStream in =null;finalbyte buffer[]=newbyte[(int) file.length()];try{
in =newFileInputStream(file);
in.read(buffer);}catch(Exception e){
pending =newRuntimeException("Exception occured on reading file "+ file.getAbsolutePath(), e);}finally{if(in !=null){try{
in.close();}catch(Exception e){if(pending ==null){
pending =newRuntimeException("Exception occured on closing file"+ file.getAbsolutePath(), e);}}}if(pending !=null){thrownewRuntimeException(pending);}}return buffer;}
не используйте пустотелые блоки это затрудняет отладку.
Sapphire_Brick
1
//The file that you wanna convert into byte[]File file=newFile("/storage/0CE2-EA3D/DCIM/Camera/VID_20190822_205931.mp4");FileInputStream fileInputStream=newFileInputStream(file);byte[] data=newbyte[(int) file.length()];BufferedInputStream bufferedInputStream=newBufferedInputStream(fileInputStream);
bufferedInputStream.read(data,0,data.length);//Now the bytes of the file are contain in the "byte[] data"
Хотя этот код может обеспечить решение вопроса, лучше добавить контекст относительно того, почему / как он работает. Это может помочь будущим пользователям учиться и применять эти знания в своем собственном коде. Вы также, вероятно, получите положительные отзывы от пользователей в виде откликов, когда код объясняется.
Борчвм
Ну, это важная часть, о которой я буду помнить в будущих постах. Спасибо за ваши полезные идеи.
Ответы:
Это зависит от того, что лучше для вас значит. Производительность мудрая, не изобретайте велосипед и используйте Apache Commons. Который здесь
IOUtils.toByteArray(InputStream input)
.источник
File
чтобыbyte[]
? Я использую Java6, поэтому я не могу использовать методы NIO :(С JDK 7 вы можете использовать
Files.readAllBytes(Path)
.Пример:
источник
Начиная с JDK 7 - один лайнер:
Внешние зависимости не нужны.
источник
import java.nio.file.Files;
иimport java.nio.file.Paths;
если вы еще не сделали.Документация для Java 8: http://docs.oracle.com/javase/8/docs/api/java/io/RandomAccessFile.html
источник
В основном вы должны прочитать это в памяти. Откройте файл, выделите массив и прочитайте содержимое файла в массив.
Самый простой способ - это что-то похожее на это:
Это приводит к некоторому ненужному копированию содержимого файла (фактически данные копируются три раза: из файла в
buffer
, из,buffer
вByteArrayOutputStream
,ByteArrayOutputStream
в реальный результирующий массив).Вам также необходимо убедиться, что вы читаете в памяти только файлы определенного размера (обычно это зависит от приложения) :-).
Вы также должны относиться к
IOException
внешней функции.Другой способ заключается в следующем:
Это не имеет ненужного копирования.
FileTooBigException
исключение пользовательского приложения.MAX_FILE_SIZE
Константа AN параметры приложения.Для больших файлов вы, вероятно, должны подумать об алгоритме обработки потока или использовать отображение памяти (см.
java.nio
).источник
Как кто-то сказал, Apache Commons File Utils может иметь то, что вы ищете
Пример использования (
Program.java
):источник
Вы можете использовать API NIO, чтобы сделать это. Я мог бы сделать это с этим кодом до тех пор, пока общий размер файла (в байтах) поместится в int.
Я думаю, что это очень быстро, так как использует MappedByteBuffer.
источник
Если у вас нет Java 8, и вы согласны со мной, что включение массивной библиотеки во избежание написания нескольких строк кода - плохая идея:
Абонент отвечает за закрытие потока.
источник
источник
throw new IOException("File is too large!");
что делать, если файл слишком большой? Есть ли еще пример?Простой способ сделать это:
источник
Самый простой способ чтения байтов из файла
источник
.*
, это считается плохой практикой.У Guava есть Files.toByteArray (), чтобы предложить вам. У этого есть несколько преимуществ:
источник
источник
Используя тот же подход, что и вики-ответ сообщества, но более чистый и компилируемый из коробки (предпочтительный подход, если вы не хотите импортировать библиотеки Apache Commons, например, на Android):
источник
Я считаю, что это самый простой способ:
источник
ReadFully Считывает байты b.length из этого файла в байтовый массив, начиная с текущего указателя файла. Этот метод читает несколько раз из файла, пока не будет прочитано запрошенное количество байтов. Этот метод блокируется до тех пор, пока не будет прочитано запрошенное количество байтов, не будет обнаружен конец потока или не сгенерировано исключение.
источник
Если вы хотите прочитать байты в заранее выделенный байтовый буфер, этот ответ может помочь.
Ваше первое предположение, вероятно, будет использовать
InputStream read(byte[])
. Однако этот метод имеет недостаток, который делает его неоправданно сложным: нет никакой гарантии, что массив будет фактически полностью заполнен, даже если EOF не обнаружен.Вместо этого взгляните на
DataInputStream readFully(byte[])
. Это оболочка для входных потоков, которая не имеет вышеуказанной проблемы. Кроме того, этот метод выбрасывает при обнаружении EOF. Гораздо приятнее.источник
Мало того, что следующий способ преобразует файл java.io.File в байт [], я также обнаружил, что это самый быстрый способ чтения в файле при тестировании множества различных методов чтения Java-файлов. друг против друга:
java.nio.file.Files.readAllBytes ()
источник
Позвольте мне добавить другое решение без использования сторонних библиотек. Он повторно использует шаблон обработки исключений, который был предложен Скоттом ( ссылка ). И я переместил некрасивую часть в отдельное сообщение (я бы спрятался в каком-то классе FileUtils;))
источник
источник
Другой способ чтения байтов из файла
источник
источник
Попробуй это :
источник
В JDK8
источник