Как читать / конвертировать InputStream в String в Java?

Если у вас есть объект java.io.InputStream , как вы должны обработать этот объект и создать String ?


Предположим, у меня есть InputStream , содержащий текстовые данные, и я хочу преобразовать его в String , так что, например, я могу записать это в файл журнала.

Какой самый простой способ взять InputStream и преобразовать его в String ?

 public String convertStreamToString(InputStream is) { 
    // ???
}
 
вопрос задан 21.11.2008
Johnny Maelstrom
17099 репутация

57 ответов


  • 2118 рейтинг

    Вот способ использования только стандартной библиотеки Java (обратите внимание, что поток не закрыт, YMMV).

     static String convertStreamToString(java.io.InputStream is) {
        java.util.Scanner s = new java.util.Scanner(is).useDelimiter("\\A");
        return s.hasNext() ? s.next() : "";
    }
     

    Я узнал этот трюк из статьи «Stupid Scanner tricks» . Причина, по которой он работает, заключается в том, что Scanner выполняет итерацию по токенам в потоке, и в этом случае мы разделяем токены, используя «начало границы ввода» (\ A), что дает нам только один токен для всего содержимого потока.

    Обратите внимание, что если вам необходимо указать кодировку входного потока, вы можете предоставить второй аргумент Scanner конструктору, который указывает, какую кодировку использовать (например, «UTF-8»).

    Кончик шляпы идет и к Иакову, который однажды указал мне на указанную статью.

    EDITED: благодаря предложению Патрика , эта функция стала более надежной при обработке пустого потока ввода. Еще одно редактирование: смешение try / catch, путь Патрика более лаконичен.

    ответ дан Pavel Repin, с репутацией 26392, 26.03.2011
  • 2085 рейтинг

    Хороший способ сделать это - использовать Apache commons IOUtils для копирования InputStream в StringWriter ... что-то вроде

     StringWriter writer = new StringWriter();
    IOUtils.copy(inputStream, writer, encoding);
    String theString = writer.toString();
     

    или даже

     // NB: does not close inputStream, you'll have to use try-with-resources for that
    String theString = IOUtils.toString(inputStream, encoding); 
     

    В качестве альтернативы вы можете использовать ByteArrayOutputStream если вы не хотите смешивать свои потоки и писатели

    ответ дан Harry Lime, с репутацией 21954, 21.11.2008
  • 1738 рейтинг

    Подводя итог другим ответам, я нашел 11 основных способов сделать это (см. Ниже). И я написал некоторые тесты производительности (см. Результаты ниже):

    Способы преобразования InputStream в строку:

    1. Использование IOUtils.toString (Apache Utils)

       String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8);
       
    2. Использование CharStreams (Гуава)

       String result = CharStreams.toString(new InputStreamReader(
            inputStream, Charsets.UTF_8));
       
    3. Использование Scanner (JDK)

       Scanner s = new Scanner(inputStream).useDelimiter("\\A");
      String result = s.hasNext() ? s.next() : "";
       
    4. Использование Stream API (Java 8). Предупреждение . Это решение преобразует разрывы строк (например, \r\n ) в \n .

       String result = new BufferedReader(new InputStreamReader(inputStream))
        .lines().collect(Collectors.joining("\n"));
       
    5. Использование параллельного Stream API (Java 8). Предупреждение . Это решение преобразует различные разрывы строк (например, \r\n ) в String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); .

       String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8);
       
    6. Использование String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); и String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); (JDK)

       String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8);
       
    7. Использование String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); и String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); (Apache Commons)

       String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8);
       
    8. Использование String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); и String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8); (JDK)

       CharStreams 
    9. Использование CharStreams (JDK). Внимание: Это решение преобразует различные разрывы строк (например , CharStreams ) до CharStreams системы собственности (например, в Windows , чтобы «\ г \ п»).

       CharStreams 
    10. Использование CharStreams и CharStreams (JDK)

       CharStreams 
    11. Использование CharStreams и CharStreams (JDK). Предупреждение . Это решение имеет проблемы с Unicode, например, с русским текстом (работает корректно только с текстом, отличным от Unicode)

       String result = CharStreams.toString(new InputStreamReader(
            inputStream, Charsets.UTF_8));
       

    Предупреждение :

    1. Решения 4, 5 и 9 конвертируют разные разрывы строк в один.

    2. Решение 11 не может работать корректно с текстом Unicode

    Тесты производительности

    Тесты производительности для малых String result = CharStreams.toString(new InputStreamReader( inputStream, Charsets.UTF_8)); (длина = 175), url в github (режим = Среднее время, система = Linux, оценка 1,343 - лучшая):

     String result = CharStreams.toString(new InputStreamReader(
          inputStream, Charsets.UTF_8));
     

    Тесты производительности для большого String result = CharStreams.toString(new InputStreamReader( inputStream, Charsets.UTF_8)); (длина = 50100), url в github (режим = Среднее время, система = Linux, оценка 200 715 - лучшая):

     String result = CharStreams.toString(new InputStreamReader(
          inputStream, Charsets.UTF_8));
     

    Графики (тесты производительности в зависимости от длины входного потока в системе Windows 7)
    введите описание изображения здесь

    Тест производительности (среднее время) в зависимости от длины входного потока в системе Windows 7:

     String result = CharStreams.toString(new InputStreamReader(
          inputStream, Charsets.UTF_8));
     
    ответ дан Viacheslav Vedenin, с репутацией 30141, 17.02.2016
  • 796 рейтинг

    Apache Commons позволяет:

     String myString = IOUtils.toString(myInputStream, "UTF-8");
     

    Конечно, вы можете выбирать другие кодировки символов, кроме UTF-8.

    Также см .: ( Документы )

    ответ дан Chinnery, с репутацией 8784, 8.12.2008
  • 265 рейтинг

    Принимая во внимание файл, сначала нужно получить экземпляр java.io.Reader . Затем это можно прочитать и добавить в StringBuilder (нам не нужно StringBuffer если мы не обращаемся к нему в нескольких потоках, а StringBuilder быстрее). Трюк здесь заключается в том, что мы работаем в блоках и, как таковые, не нуждаемся в других буферизационных потоках. Размер блока параметризуется для оптимизации производительности во время выполнения.

     public static String slurp(final InputStream is, final int bufferSize) {
        final char[] buffer = new char[bufferSize];
        final StringBuilder out = new StringBuilder();
        try (Reader in = new InputStreamReader(is, "UTF-8")) {
            for (;;) {
                int rsz = in.read(buffer, 0, buffer.length);
                if (rsz < 0)
                    break;
                out.append(buffer, 0, rsz);
            }
        }
        catch (UnsupportedEncodingException ex) {
            /* ... */
        }
        catch (IOException ex) {
            /* ... */
        }
        return out.toString();
    }
     
    ответ дан Paul de Vrieze, с репутацией 4539, 21.11.2008
  • 228 рейтинг

    Как насчет этого?

     InputStream in = /* your InputStream */;
    StringBuilder sb=new StringBuilder();
    BufferedReader br = new BufferedReader(new InputStreamReader(in));
    String read;
    
    while((read=br.readLine()) != null) {
        //System.out.println(read);
        sb.append(read);   
    }
    
    br.close();
    return sb.toString();
     
    ответ дан sampathpremarathna, с репутацией 3253, 4.08.2011
  • 154 рейтинг

    Если вы используете Google-Collections / Guava, вы можете сделать следующее:

     InputStream stream = ...
    String content = CharStreams.toString(new InputStreamReader(stream, Charsets.UTF_8));
    Closeables.closeQuietly(stream);
     

    Обратите внимание, что второй параметр (т.е. Charsets.UTF_8) для InputStreamReader не нужен, но, как правило, рекомендуется указывать кодировку, если вы ее знаете (что вам нужно!)

    ответ дан Sakuraba, с репутацией 2208, 13.07.2010
  • 108 рейтинг

    Это мое чистое решение для Java и Android, хорошо работает ...

     public String readFullyAsString(InputStream inputStream, String encoding)
            throws IOException {
        return readFully(inputStream).toString(encoding);
    }    
    
    public byte[] readFullyAsBytes(InputStream inputStream)
            throws IOException {
        return readFully(inputStream).toByteArray();
    }    
    
    private ByteArrayOutputStream readFully(InputStream inputStream)
            throws IOException {
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        byte[] buffer = new byte[1024];
        int length = 0;
        while ((length = inputStream.read(buffer)) != -1) {
            baos.write(buffer, 0, length);
        }
        return baos;
    }
     
    ответ дан TacB0sS, с репутацией 5555, 8.05.2012
  • 56 рейтинг

    Как насчет:

     import java.io.BufferedInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.InputStream;
    import java.io.IOException;    
    
    public static String readInputStreamAsString(InputStream in) 
        throws IOException {
    
        BufferedInputStream bis = new BufferedInputStream(in);
        ByteArrayOutputStream buf = new ByteArrayOutputStream();
        int result = bis.read();
        while(result != -1) {
          byte b = (byte)result;
          buf.write(b);
          result = bis.read();
        }        
        return buf.toString();
    }
     
    ответ дан Jon Moore, с репутацией 1114, 10.06.2009
  • 55 рейтинг

    Вот самое изящное, чисто-Java (без библиотеки) решение, которое я придумал после некоторых экспериментов:

     public static String fromStream(InputStream in) throws IOException
    {
        BufferedReader reader = new BufferedReader(new InputStreamReader(in));
        StringBuilder out = new StringBuilder();
        String newLine = System.getProperty("line.separator");
        String line;
        while ((line = reader.readLine()) != null) {
            out.append(line);
            out.append(newLine);
        }
        return out.toString();
    }
     
    ответ дан Drew Noakes, с репутацией 178444, 1.01.2013
  • 43 рейтинг

    Для полноты здесь решение Java 9 :

     public static String toString(InputStream input) throws IOException {
        return new String(input.readAllBytes(), StandardCharsets.UTF_8);
    }
     

    readAllBytes в настоящее время находится в основной базе JDK 9, поэтому он, вероятно, появится в релизе. Вы можете попробовать прямо сейчас, используя сборки моментальных снимков JDK 9 .

    ответ дан Tagir Valeev, с репутацией 65050, 2.09.2015
  • 33 рейтинг

    Я бы использовал некоторые трюки Java 8.

     public static String streamToString(final InputStream inputStream) throws Exception {
        // buffering optional
        try
        (
            final BufferedReader br
               = new BufferedReader(new InputStreamReader(inputStream))
        ) {
            // parallel optional
            return br.lines().parallel().collect(Collectors.joining("\n"));
        } catch (final IOException e) {
            throw new RuntimeException(e);
            // whatever.
        }
    }
     

    По сути, такие же, как некоторые другие ответы, кроме более кратким.

    ответ дан Simon Kuang, с репутацией 2192, 17.07.2014
  • 27 рейтинг

    Я провел несколько тестов времени, потому что время имеет значение, всегда.

    Я попытался получить ответ в String 3 разными способами. (показано ниже)
    Я оставил блоки try / catch для удобства чтения.

    Чтобы дать контекст, это предыдущий код для всех трех подходов:

        String response;
       String url = "www.blah.com/path?key=value";
       GetMethod method = new GetMethod(url);
       int status = client.executeMethod(method);
     

    1)

      response = method.getResponseBodyAsString();
     

    2)

     InputStream resp = method.getResponseBodyAsStream();
    InputStreamReader is=new InputStreamReader(resp);
    BufferedReader br=new BufferedReader(is);
    String read = null;
    StringBuffer sb = new StringBuffer();
    while((read = br.readLine()) != null) {
        sb.append(read);
    }
    response = sb.toString();
     

    3)

     InputStream iStream  = method.getResponseBodyAsStream();
    StringWriter writer = new StringWriter();
    IOUtils.copy(iStream, writer, "UTF-8");
    response = writer.toString();
     

    Таким образом, после выполнения 500 тестов для каждого подхода с теми же данными запроса / ответа, вот цифры. Еще раз, это мои выводы, и ваши результаты могут быть не совсем одинаковыми, но я написал это, чтобы дать некоторые указания другим относительно различий в эффективности этих подходов.

    Звания:
    Подход №1
    Подход № 3 - на 2.6% медленнее, чем # 1
    Подход №2 - на 4,3% медленнее, чем №1

    Любой из этих подходов является подходящим решением для захвата ответа и создания из него String.

    ответ дан Brett Holt, с репутацией 995, 12.10.2011
  • 24 рейтинг

    Я сделал оценку на 14 различных ответах здесь (Извините за отсутствие кредитов, но слишком много дубликатов)

    Результат очень удивителен. Оказывается, что Apache IOUtils является самым медленным, а ByteArrayOutputStream - самым быстрым решением:

    Итак, вот лучший способ:

     public String inputStreamToString(InputStream inputStream) throws IOException {
        try(ByteArrayOutputStream result = new ByteArrayOutputStream()) {
            byte[] buffer = new byte[1024];
            int length;
            while ((length = inputStream.read(buffer)) != -1) {
                result.write(buffer, 0, length);
            }
    
            return result.toString(UTF_8);
        }
    }
     

    Результаты тестов, из 20 Мбайт случайных байтов в 20 циклов

    Время в миллисекундах

    • ByteArrayOutputStreamTest: 194
    • NioStream: 198
    • Java9ISTransferTo: 201
    • Java9ISReadAllBytes: 205
    • BufferedInputStreamVsByteArrayOutputStream: 314
    • ApacheStringWriter2: 574
    • GuavaCharStreams: 589
    • ScannerReaderNoNextTest: 614
    • ScannerReader: 633
    • ApacheStringWriter: 1544
    • StreamApi: ошибка
    • ParallelStreamApi: ошибка
    • BufferReaderTest: ошибка
    • InputStreamAndStringBuilder: ошибка

    Исходный код

     import com.google.common.io.CharStreams;
    import org.apache.commons.io.IOUtils;
    
    import java.io.*;
    import java.nio.ByteBuffer;
    import java.nio.channels.Channels;
    import java.nio.channels.ReadableByteChannel;
    import java.nio.channels.WritableByteChannel;
    import java.util.Arrays;
    import java.util.List;
    import java.util.Random;
    import java.util.stream.Collectors;
    
    /**
     * Created by Ilya Gazman on 2/13/18.
     */
    public class InputStreamToString {
    
    
        private static final String UTF_8 = "UTF-8";
    
        public static void main(String... args) {
            log("App started");
            byte[] bytes = new byte[1024 * 1024];
            new Random().nextBytes(bytes);
            log("Stream is ready\n");
    
            try {
                test(bytes);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    
        private static void test(byte[] bytes) throws IOException {
            List<Stringify> tests = Arrays.asList(
                    new ApacheStringWriter(),
                    new ApacheStringWriter2(),
                    new NioStream(),
                    new ScannerReader(),
                    new ScannerReaderNoNextTest(),
                    new GuavaCharStreams(),
                    new StreamApi(),
                    new ParallelStreamApi(),
                    new ByteArrayOutputStreamTest(),
                    new BufferReaderTest(),
                    new BufferedInputStreamVsByteArrayOutputStream(),
                    new InputStreamAndStringBuilder(),
                    new Java9ISTransferTo(),
                    new Java9ISReadAllBytes()
            );
    
            String solution = new String(bytes, "UTF-8");
    
            for (Stringify test : tests) {
                try (ByteArrayInputStream inputStream = new ByteArrayInputStream(bytes)) {
                    String s = test.inputStreamToString(inputStream);
                    if (!s.equals(solution)) {
                        log(test.name() + ": Error");
                        continue;
                    }
                }
                long startTime = System.currentTimeMillis();
                for (int i = 0; i < 20; i++) {
                    try (ByteArrayInputStream inputStream = new ByteArrayInputStream(bytes)) {
                        test.inputStreamToString(inputStream);
                    }
                }
                log(test.name() + ": " + (System.currentTimeMillis() - startTime));
            }
        }
    
        private static void log(String message) {
            System.out.println(message);
        }
    
        interface Stringify {
            String inputStreamToString(InputStream inputStream) throws IOException;
    
            default String name() {
                return this.getClass().getSimpleName();
            }
        }
    
        static class ApacheStringWriter implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                StringWriter writer = new StringWriter();
                IOUtils.copy(inputStream, writer, UTF_8);
                return writer.toString();
            }
        }
    
        static class ApacheStringWriter2 implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                return IOUtils.toString(inputStream, UTF_8);
            }
        }
    
        static class NioStream implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream in) throws IOException {
                ReadableByteChannel channel = Channels.newChannel(in);
                ByteBuffer byteBuffer = ByteBuffer.allocate(1024 * 16);
                ByteArrayOutputStream bout = new ByteArrayOutputStream();
                WritableByteChannel outChannel = Channels.newChannel(bout);
                while (channel.read(byteBuffer) > 0 || byteBuffer.position() > 0) {
                    byteBuffer.flip();  //make buffer ready for write
                    outChannel.write(byteBuffer);
                    byteBuffer.compact(); //make buffer ready for reading
                }
                channel.close();
                outChannel.close();
                return bout.toString(UTF_8);
            }
        }
    
        static class ScannerReader implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream is) throws IOException {
                java.util.Scanner s = new java.util.Scanner(is).useDelimiter("\\A");
                return s.hasNext() ? s.next() : "";
            }
        }
    
        static class ScannerReaderNoNextTest implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream is) throws IOException {
                java.util.Scanner s = new java.util.Scanner(is).useDelimiter("\\A");
                return s.next();
            }
        }
    
        static class GuavaCharStreams implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream is) throws IOException {
                return CharStreams.toString(new InputStreamReader(
                        is, UTF_8));
            }
        }
    
        static class StreamApi implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                return new BufferedReader(new InputStreamReader(inputStream))
                        .lines().collect(Collectors.joining("\n"));
            }
        }
    
        static class ParallelStreamApi implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                return new BufferedReader(new InputStreamReader(inputStream)).lines()
                        .parallel().collect(Collectors.joining("\n"));
            }
        }
    
        static class ByteArrayOutputStreamTest implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                try(ByteArrayOutputStream result = new ByteArrayOutputStream()) {
                    byte[] buffer = new byte[1024];
                    int length;
                    while ((length = inputStream.read(buffer)) != -1) {
                        result.write(buffer, 0, length);
                    }
    
                    return result.toString(UTF_8);
                }
            }
        }
    
        static class BufferReaderTest implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                String newLine = System.getProperty("line.separator");
                BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
                StringBuilder result = new StringBuilder(UTF_8);
                String line;
                boolean flag = false;
                while ((line = reader.readLine()) != null) {
                    result.append(flag ? newLine : "").append(line);
                    flag = true;
                }
                return result.toString();
            }
        }
    
        static class BufferedInputStreamVsByteArrayOutputStream implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                BufferedInputStream bis = new BufferedInputStream(inputStream);
                ByteArrayOutputStream buf = new ByteArrayOutputStream();
                int result = bis.read();
                while (result != -1) {
                    buf.write((byte) result);
                    result = bis.read();
                }
    
                return buf.toString(UTF_8);
            }
        }
    
        static class InputStreamAndStringBuilder implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                int ch;
                StringBuilder sb = new StringBuilder(UTF_8);
                while ((ch = inputStream.read()) != -1)
                    sb.append((char) ch);
                return sb.toString();
            }
        }
    
        static class Java9ISTransferTo implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                ByteArrayOutputStream bos = new ByteArrayOutputStream();
                inputStream.transferTo(bos);
                return bos.toString(UTF_8);
            }
        }
    
        static class Java9ISReadAllBytes implements Stringify {
    
            @Override
            public String inputStreamToString(InputStream inputStream) throws IOException {
                return new String(inputStream.readAllBytes(), UTF_8);
            }
        }
    
    }
     
    ответ дан Ilya Gazman, с репутацией 14738, 13.02.2018
  • 24 рейтинг

    Чистое решение Java с использованием Stream s работает с Java 8.

     import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.InputStreamReader;
    import java.util.stream.Collectors;
    
    // ...
    public static String inputStreamToString(InputStream is) throws IOException {
        try (BufferedReader br = new BufferedReader(new InputStreamReader(is))) {
            return br.lines().collect(Collectors.joining(System.lineSeparator()));
        }
    }
     

    Как упоминалось Кристоффером Хаммарстремом ниже другого ответа, более безопасно явно указывать Charset . Т.е. конструктор InputStreamReader может быть изменен следующим образом:

     new InputStreamReader(is, Charset.forName("UTF-8"))
     
    ответ дан czerny, с репутацией 4396, 26.02.2015
  • 21 рейтинг

    Вот более или менее ответ sampath, немного очищенный и представленный как функция:

     String streamToString(InputStream in) throws IOException {
      StringBuilder out = new StringBuilder();
      BufferedReader br = new BufferedReader(new InputStreamReader(in));
      for(String line = br.readLine(); line != null; line = br.readLine()) 
        out.append(line);
      br.close();
      return out.toString();
    }
     
    ответ дан TKH, с репутацией 609, 30.03.2012
  • 19 рейтинг

    Если бы вы чувствовали себя авантюрно, вы могли бы смешать Scala и Java и в итоге:

     scala.io.Source.fromInputStream(is).mkString("")
     

    Смешивание кода Java и Scala и библиотек имеет свои преимущества.

    См. Полное описание здесь: Идиоматический способ преобразования InputStream в строку в Scala

    ответ дан Jack, с репутацией 9309, 7.03.2012
  • 17 рейтинг

    Если вы не можете использовать Commons IO (FileUtils / IOUtils / CopyUtils), вот пример использования BufferedReader для чтения файла по строкам:

     public class StringFromFile {
        public static void main(String[] args) /*throws UnsupportedEncodingException*/ {
            InputStream is = StringFromFile.class.getResourceAsStream("file.txt");
            BufferedReader br = new BufferedReader(new InputStreamReader(is/*, "UTF-8"*/));
            final int CHARS_PER_PAGE = 5000; //counting spaces
            StringBuilder builder = new StringBuilder(CHARS_PER_PAGE);
            try {
                for(String line=br.readLine(); line!=null; line=br.readLine()) {
                    builder.append(line);
                    builder.append('\n');
                }
            } catch (IOException ignore) { }
            String text = builder.toString();
            System.out.println(text);
        }
    }
     

    или, если вы хотите получить необработанную скорость, я бы предложил вариант, предложенный Paul De Vrieze (который избегает использования StringWriter (который использует StringBuffer внутри):

     public class StringFromFileFast {
        public static void main(String[] args) /*throws UnsupportedEncodingException*/ {
            InputStream is = StringFromFileFast.class.getResourceAsStream("file.txt");
            InputStreamReader input = new InputStreamReader(is/*, "UTF-8"*/);
            final int CHARS_PER_PAGE = 5000; //counting spaces
            final char[] buffer = new char[CHARS_PER_PAGE];
            StringBuilder output = new StringBuilder(CHARS_PER_PAGE);
            try {
                for(int read = input.read(buffer, 0, buffer.length);
                        read != -1;
                        read = input.read(buffer, 0, buffer.length)) {
                    output.append(buffer, 0, read);
                }
            } catch (IOException ignore) { }
    
            String text = output.toString();
            System.out.println(text);
        }
    }
     
    ответ дан DJDaveMark, с репутацией 1313, 18.05.2010
  • 16 рейтинг

    Не забудьте закрыть потоки в конце, если вы используете Stream Readers

     private String readStream(InputStream iStream) throws IOException {
        //build a Stream Reader, it can read char by char
        InputStreamReader iStreamReader = new InputStreamReader(iStream);
        //build a buffered Reader, so that i can read whole line at once
        BufferedReader bReader = new BufferedReader(iStreamReader);
        String line = null;
        StringBuilder builder = new StringBuilder();
        while((line = bReader.readLine()) != null) {  //Read till end
            builder.append(line);
            builder.append("\n"); // append new line to preserve lines
        }
        bReader.close();         //close all opened stuff
        iStreamReader.close();
        //iStream.close(); //EDIT: Let the creator of the stream close it!
                           // some readers may auto close the inner stream
        return builder.toString();
    }
     

    EDIT: в JDK 7+ вы можете использовать конструкцию try-with-resources.

     /**
     * Reads the stream into a string
     * @param iStream the input stream
     * @return the string read from the stream
     * @throws IOException when an IO error occurs
     */
    private String readStream(InputStream iStream) throws IOException {
    
        //Buffered reader allows us to read line by line
        try (BufferedReader bReader =
                     new BufferedReader(new InputStreamReader(iStream))){
            StringBuilder builder = new StringBuilder();
            String line;
            while((line = bReader.readLine()) != null) {  //Read till end
                builder.append(line);
                builder.append("\n"); // append new line to preserve lines
            }
            return builder.toString();
        }
    }
     
    ответ дан Thamme Gowda, с репутацией 5894, 17.11.2012
  • 16 рейтинг

    Это ответ адаптирован из org.apache.commons.io.IOUtils исходного кода , для тех , кто хочет иметь реализацию апачской , но не хочет целую библиотеки.

     private static final int BUFFER_SIZE = 4 * 1024;
    
    public static String inputStreamToString(InputStream inputStream, String charsetName)
            throws IOException {
        StringBuilder builder = new StringBuilder();
        InputStreamReader reader = new InputStreamReader(inputStream, charsetName);
        char[] buffer = new char[BUFFER_SIZE];
        int length;
        while ((length = reader.read(buffer)) != -1) {
            builder.append(buffer, 0, length);
        }
        return builder.toString();
    }
     
    ответ дан Dreaming in Code, с репутацией 2406, 3.08.2014
  • 14 рейтинг

    Вот полный метод преобразования InputStream в String без использования сторонней библиотеки. Используйте StringBuilder для однопоточной среды, иначе используйте StringBuffer .

     public static String getString( InputStream is) throws IOException {
        int ch;
        StringBuilder sb = new StringBuilder();
        while((ch = is.read()) != -1)
            sb.append((char)ch);
        return sb.toString();
    }
     
    ответ дан laksys, с репутацией 2218, 9.04.2014
  • 13 рейтинг

    Вот как это сделать, используя только JDK с использованием буферов массива байтов. Это фактически то, как все методы commons-io IOUtils.copy() работают. Вы можете заменить byte[] на char[] если вы копируете из Reader вместо InputStream .

     import java.io.ByteArrayOutputStream;
    import java.io.InputStream;
    
    ...
    
    InputStream is = ....
    ByteArrayOutputStream baos = new ByteArrayOutputStream(8192);
    byte[] buffer = new byte[8192];
    int count = 0;
    try {
      while ((count = is.read(buffer)) != -1) {
        baos.write(buffer, 0, count);
      }
    }
    finally {
      try {
        is.close();
      }
      catch (Exception ignore) {
      }
    }
    
    String charset = "UTF-8";
    String inputStreamAsString = baos.toString(charset);
     
    ответ дан Matt Shannon, с репутацией 139, 2.11.2012
  • 11 рейтинг

    Еще один, для всех пользователей Spring:

     import java.nio.charset.StandardCharsets;
    import org.springframework.util.FileCopyUtils;
    
    public String convertStreamToString(InputStream is) throws IOException { 
        return new String(FileCopyUtils.copyToByteArray(is), StandardCharsets.UTF_8);
    }
     

    Методы утилиты в org.springframework.util.StreamUtils аналогичны методам в FileCopyUtils , но они оставляют поток открытым, когда это делается.

    ответ дан James, с репутацией 6783, 29.07.2016
  • 11 рейтинг

    Используйте java.io.InputStream.transferTo (OutputStream), поддерживаемый в Java 9, и ByteArrayOutputStream.toString (String), который берет имя кодировки:

     public static String gobble(InputStream in, String charsetName) throws IOException {
        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        in.transferTo(bos);
        return bos.toString(charsetName);
    }
     
    ответ дан jmehrens, с репутацией 6441, 28.01.2016
  • 11 рейтинг

    Пользователи Kotlin просто делают:

     println(InputStreamReader(is).readText())
     

    в то время как

     readText()
     

    это встроенный метод расширения стандартной библиотеки Kotlin.

    ответ дан Alex, с репутацией 3977, 4.02.2015
  • 9 рейтинг

    Это хорошо, потому что:

    • Ручная безопасность Charset.
    • Вы контролируете размер буфера чтения.
    • Вы можете указать длину строителя и быть не совсем точным.
    • Является свободным от библиотечных зависимостей.
    • Является ли для Java 7 или выше.

    Для чего?

     public static String convertStreamToString(InputStream is) {
       if (is == null) return null;
       StringBuilder sb = new StringBuilder(2048); // Define a size if you have an idea of it.
       char[] read = new char[128]; // Your buffer size.
       try (InputStreamReader ir = new InputStreamReader(is, StandardCharsets.UTF_8)) {
         for (int i; -1 != (i = ir.read(read)); sb.append(read, 0, i));
       } catch (Throwable t) {}
       return sb.toString();
    }
     
    ответ дан Daniel De León, с репутацией 8786, 8.06.2014
  • 6 рейтинг

    Самый простой способ в JDK состоит в следующем: snipplets кода.

     String convertToString(InputStream in){
        String resource = new Scanner(in).useDelimiter("\\Z").next();
        return resource;
    }
     
    ответ дан Raghu K Nair, с репутацией 2279, 9.08.2016
  • 6 рейтинг

    Вот мое решение на базе Java 8 , которое использует новый Stream API для сбора всех строк из InputStream :

     public static String toString(InputStream inputStream) {
        BufferedReader reader = new BufferedReader(
            new InputStreamReader(inputStream));
        return reader.lines().collect(Collectors.joining(
            System.getProperty("line.separator")));
    }
     
    ответ дан Christian Rädel, с репутацией 291, 2.09.2015
  • 4 рейтинг

    В Groovy

     inputStream.getText()
     
    ответ дан Snekse, с репутацией 10500, 26.01.2017
  • 4 рейтинг

    В терминах reduce и concat он может быть выражен в Java 8 как:

     String fromFile = new BufferedReader(new   
    InputStreamReader(inputStream)).lines().reduce(String::concat).get();
     
    ответ дан libnull-dev, с репутацией 437, 21.01.2016
  • 4 рейтинг

    Примечание. Это, вероятно, не очень хорошая идея. Этот метод использует рекурсию:

     public String read (InputStream is) {
        byte next = is.read();
        return next == -1 ? "" : next + read(is); // Recursive part: reads next byte recursively
    }
     

    Пожалуйста, не уменьшайте это только потому, что это плохой выбор; это было главным образом творческим :)

    ответ дан HyperNeutrino, с репутацией 3273, 17.11.2015
  • 4 рейтинг

    Raghu K Nair Был единственным, кто использовал сканер. Код, который я использую, немного отличается:

     String convertToString(InputStream in){
        Scanner scanner = new Scanner(in)
        scanner.useDelimiter("\\A");
    
        boolean hasInput = scanner.hasNext();
        if (hasInput) {
            return scanner.next();
        } else {
            return null;
        }
    
    }
     

    О разделителях: как использовать разделитель в сканере Java?

    ответ дан Halfacht, с репутацией 88, 10.05.2017
  • 4 рейтинг

    Следующий код работал для меня.

     URL url = MyClass.class.getResource("/" + configFileName);
    BufferedInputStream bi = (BufferedInputStream) url.getContent();
    byte[] buffer = new byte[bi.available() ];
    int bytesRead = bi.read(buffer);
    String out = new String(buffer);
     

    Обратите внимание: согласно документам Java, метод available() может работать не с InputStream но всегда работает с BufferedInputStream . Если вы не хотите использовать метод available() мы всегда можем использовать приведенный ниже код

     URL url = MyClass.class.getResource("/" + configFileName);
    BufferedInputStream bi = (BufferedInputStream) url.getContent();
    File f = new File(url.getPath());
    byte[] buffer = new byte[ (int) f.length()];
    int bytesRead = bi.read(buffer);
    String out = new String(buffer);
     

    Я не уверен, будут ли какие-либо проблемы с кодировкой. Прокомментируйте, если возникнут какие-либо проблемы с кодом.

    ответ дан Anand N, с репутацией 167, 24.07.2012
  • 4 рейтинг

    Guava предоставляет гораздо более короткое эффективное решение для автоматической блокировки, если поток ввода поступает из ресурса classpath (который, кажется, является популярной задачей):

     byte[] bytes = Resources.toByteArray(classLoader.getResource(path));
     

    или

     String text = Resources.toString(classLoader.getResource(path), StandardCharsets.UTF_8);
     

    Существует также общая концепция ByteSource и CharSource, которые осторожно заботятся об открытии и закрытии потока.

    Так, например, вместо явного открытия небольшого файла для чтения его содержимого:

     String content = Files.asCharSource(new File("robots.txt"), StandardCharsets.UTF_8).read();
    byte[] data = Files.asByteSource(new File("favicon.ico")).read();
     

    или просто

     String content = Files.toString(new File("robots.txt"), StandardCharsets.UTF_8);
    byte[] data = Files.toByteArray(new File("favicon.ico"));
     
    ответ дан Vadzim, с репутацией 13823, 9.07.2015
  • 3 рейтинг

    Я написал класс, который делает именно это, поэтому я решил, что поделюсь им со всеми. Иногда вы не хотите добавлять Apache Commons только для одного, и хотите что-то нелепое, чем сканер, который не проверяет содержимое.

    Использование выглядит следующим образом

     // Read from InputStream
    String data = new ReaderSink(inputStream, Charset.forName("UTF-8")).drain();
    
    // Read from File
    data = new ReaderSink(file, Charset.forName("UTF-8")).drain();
    
    // Drain input stream to console
    new ReaderSink(inputStream, Charset.forName("UTF-8")).drainTo(System.out);
     

    Вот код для ReaderSink:

     import java.io.*;
    import java.nio.charset.Charset;
    
    /**
     * A simple sink class that drains a {@link Reader} to a {@link String} or
     * to a {@link Writer}.
     *
     * @author Ben Barkay
     * @version 2/20/2014
     */
    public class ReaderSink {
        /**
         * The default buffer size to use if no buffer size was specified.
         */
        public static final int DEFAULT_BUFFER_SIZE = 1024;
    
        /**
         * The {@link Reader} that will be drained.
         */
        private final Reader in;
    
        /**
         * Constructs a new {@code ReaderSink} for the specified file and charset.
         * @param file      The file to read from.
         * @param charset   The charset to use.
         * @throws FileNotFoundException    If the file was not found on the filesystem.
         */
        public ReaderSink(File file, Charset charset) throws FileNotFoundException {
            this(new FileInputStream(file), charset);
        }
    
        /**
         * Constructs a new {@code ReaderSink} for the specified {@link InputStream}.
         * @param in        The {@link InputStream} to drain.
         * @param charset   The charset to use.
         */
        public ReaderSink(InputStream in, Charset charset) {
            this(new InputStreamReader(in, charset));
        }
    
        /**
         * Constructs a new {@code ReaderSink} for the specified {@link Reader}.
         * @param in    The reader to drain.
         */
        public ReaderSink(Reader in) {
            this.in = in;
        }
    
        /**
         * Drains the data from the underlying {@link Reader}, returning a {@link String} containing
         * all of the read information. This method will use {@link #DEFAULT_BUFFER_SIZE} for
         * its buffer size.
         * @return  A {@link String} containing all of the information that was read.
         */
        public String drain() throws IOException {
            return drain(DEFAULT_BUFFER_SIZE);
        }
    
        /**
         * Drains the data from the underlying {@link Reader}, returning a {@link String} containing
         * all of the read information.
         * @param bufferSize    The size of the buffer to use when reading.
         * @return  A {@link String} containing all of the information that was read.
         */
        public String drain(int bufferSize) throws IOException {
            StringWriter stringWriter = new StringWriter();
            drainTo(stringWriter, bufferSize);
            return stringWriter.toString();
        }
    
        /**
         * Drains the data from the underlying {@link Reader}, writing it to the
         * specified {@link Writer}. This method will use {@link #DEFAULT_BUFFER_SIZE} for
         * its buffer size.
         * @param out   The {@link Writer} to write to.
         */
        public void drainTo(Writer out) throws IOException {
            drainTo(out, DEFAULT_BUFFER_SIZE);
        }
    
        /**
         * Drains the data from the underlying {@link Reader}, writing it to the
         * specified {@link Writer}.
         * @param out           The {@link Writer} to write to.
         * @param bufferSize    The size of the buffer to use when reader.
         */
        public void drainTo(Writer out, int bufferSize) throws IOException {
            char[] buffer = new char[bufferSize];
            int read;
            while ((read = in.read(buffer)) > -1) {
                out.write(buffer, 0, read);
            }
        }
    }
     
    ответ дан Ben Barkay, с репутацией 4136, 20.02.2014
  • 3 рейтинг
    public String read(InputStream in) throws IOException {
        try (BufferedReader buffer = new BufferedReader(new InputStreamReader(in))) {
            return buffer.lines().collect(Collectors.joining("\n"));
        }
    }
    
    ответ дан Hao Zheng, с репутацией 99, 9.01.2017
  • 3 рейтинг

    Ответ JDK 7/8, который закрывает поток и по-прежнему вызывает исключение IOException:

     StringBuilder build = new StringBuilder();
    byte[] buf = new byte[1024];
    int length;
    try (InputStream is = getInputStream()) {
      while ((length = is.read(buf)) != -1) {
        build.append(new String(buf, 0, length));
      }
    }
     
    ответ дан Brian Pontarelli, с репутацией 604, 5.12.2013
  • 3 рейтинг

    Основываясь на второй части принятого ответа Apache Commons, но с небольшим зазором, заполненным для всегда закрытия потока:

         String theString;
        try {
            theString = IOUtils.toString(inputStream, encoding);
        } finally {
            IOUtils.closeQuietly(inputStream);
        }
     
    ответ дан Steve Chambers, с репутацией 20910, 24.12.2015
  • 3 рейтинг

    Вы можете использовать apache commons. В IOUtils вы можете найти метод toString с тремя полезными реализациями.

     public static String toString(InputStream input) throws IOException {
            return toString(input, Charset.defaultCharset());
    }
    
    public static String toString(InputStream input) throws IOException {
            return toString(input, Charset.defaultCharset());
    }
    
    public static String toString(InputStream input, String encoding)
                throws IOException {
            return toString(input, Charsets.toCharset(encoding));
    }
     
    ответ дан Rys, с репутацией 1111, 16.01.2014
  • 3 рейтинг
    InputStream is = Context.openFileInput(someFileName); // whatever format you have
    
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    
    byte[] b = new byte[8192];
    for (int bytesRead; (bytesRead = is.read(b)) != -1;) {
        bos.write(b, 0, bytesRead);
    }
    
    String output = bos.toString(someEncoding);
    
    ответ дан Vaishali Sutariya, с репутацией 3608, 25.08.2014
  • 3 рейтинг
     InputStreamReader i = new InputStreamReader(s);
    BufferedReader str = new BufferedReader(i);
    String msg = str.readLine();
    System.out.println(msg);
     

    Здесь s - ваш InputStream объект, который преобразуется в String

    ответ дан Omkar Khot, с репутацией 71, 30.05.2013
  • 3 рейтинг

    Попробуйте эти 4 заявления.

    В соответствии с точкой, на которую ссылается Фред, не рекомендуется добавлять оператор String с += поскольку каждый раз добавляется новый char к существующим String создающим новый String объект и присваивающий его адрес st а старый st объект становится мусором ,

     public String convertStreamToString(InputStream is)
    {
        int k;
        StringBuffer sb=new StringBuffer();
        while((k=fin.read()) != -1)
        {
            sb.append((char)k);
        }
        return sb.toString();
    }
     

    Не рекомендуется, но это также способ

     public String convertStreamToString(InputStream is) { 
        int k;
        String st="";
        while((k=is.read()) != -1)
        {
            st+=(char)k;
        }
        return st;
    }
     
    ответ дан JavaTechnical, с репутацией 2182, 7.02.2014
  • 3 рейтинг

    Ну, вы можете запрограммировать это для себя .. это не сложно ..

     String Inputstream2String (InputStream is) throws IOException 
        {
            final int PKG_SIZE = 1024;
            byte[] data = new byte [PKG_SIZE];
            StringBuilder buffer = new StringBuilder(PKG_SIZE * 10);
            int size;
    
            size = is.read(data, 0, data.length);
            while (size > 0)
            {
                String str = new String(data, 0, size);
                buffer.append(str);
                size = is.read(data, 0, data.length);
            }
            return buffer.toString();
        }
     
    ответ дан Victor, с репутацией 2368, 9.03.2013
  • 2 рейтинг

    Ниже не рассматривается исходный вопрос, а скорее некоторые из ответов.

    В нескольких ответах указаны петли формы

     String line = null;
    while((line = reader.readLine()) != null) {
      // ...
    }
     

    или

     for(String line = reader.readLine(); line != null; line = reader.readLine()) {
        // ...
    }
     

    Первая форма загрязняет пространство имен охватывающей области, объявляя переменную «read» в охватывающей области, которая не будет использоваться ни для чего вне цикла for. Вторая форма дублирует вызов readline ().

    Вот гораздо более чистый способ написать такой цикл в Java. Оказывается, первое предложение в цикле for не требует действительного значения инициализатора. Это удерживает область действия переменной «строка» внутри тела цикла for. Гораздо более элегантно! Я не видел никого, кто использовал эту форму где-нибудь (я случайно обнаружил ее один день назад), но я использую ее все время.

     for (String line; (line = reader.readLine()) != null; ) {
        //...
    }
     
    ответ дан Luke Hutchison, с репутацией 2613, 11.09.2014
  • 2 рейтинг

    Вы можете использовать Cactoos :

     String text = new TextOf(inputStream).asString();
     

    Кодировка UTF-8 по умолчанию. Если вам нужен еще один:

     String text = new TextOf(inputStream, "UTF-16").asString();
     
    ответ дан yegor256, с репутацией 53954, 6.08.2017
  • 1 рейтинг

    С Окио:

     String result = Okio.buffer(Okio.source(inputStream)).readUtf8();
     
    ответ дан drakeet, с репутацией 1508, 7.04.2018
  • 1 рейтинг

    String theString = IOUtils.toString (inputStream, encoding);

    ответ дан Avinash, с репутацией 212, 20.01.2018
  • 1 рейтинг

    Это решение этого вопроса не является самым простым, но поскольку потоки и каналы NIO не упоминались, здесь идет версия, которая использует каналы NIO и ByteBuffer для преобразования потока в строку.

     public static String streamToStringChannel(InputStream in, String encoding, int bufSize) throws IOException {
        ReadableByteChannel channel = Channels.newChannel(in);
        ByteBuffer byteBuffer = ByteBuffer.allocate(bufSize);
        ByteArrayOutputStream bout = new ByteArrayOutputStream();
        WritableByteChannel outChannel = Channels.newChannel(bout);
        while (channel.read(byteBuffer) > 0 || byteBuffer.position() > 0) {
            byteBuffer.flip();  //make buffer ready for write
            outChannel.write(byteBuffer);
            byteBuffer.compact(); //make buffer ready for reading
        }
        channel.close();
        outChannel.close();
        return bout.toString(encoding);
    }
     

    Вот пример того, как его использовать:

     try (InputStream in = new FileInputStream("/tmp/large_file.xml")) {
        String x = streamToStringChannel(in, "UTF-8", 1);
        System.out.println(x);
    }
     

    Производительность этого метода должна быть хорошей для больших файлов.

    ответ дан gil.fernandes, с репутацией 5197, 21.10.2017
  • 1 рейтинг
    InputStream  inputStream = null;
    BufferedReader bufferedReader = null;
    try {
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
        String stringBuilder = new StringBuilder();
        String content;
        while((content = bufferedReader.readLine()) != null){
            stringBuilder.append(content);
        }
        System.out.println("content of file::" + stringBuilder.toString());
    }
    catch (IOException e) {
                e.printStackTrace();
            }finally{           
                if(bufferedReader != null){
                    try{
                        bufferedReader.close();
                    }catch(IoException ex){
                       ex.printStackTrace();
                }
    
    ответ дан Harsh, с репутацией 1569, 10.03.2016
  • 1 рейтинг

    Способ преобразования inputStream в String

     public static String getStringFromInputStream(InputStream inputStream) {
    
        BufferedReader bufferedReader = null;
        StringBuilder stringBuilder = new StringBuilder();
        String line;
    
        try {
            bufferedReader = new BufferedReader(new InputStreamReader(
                    inputStream));
            while ((line = bufferedReader.readLine()) != null) {
                stringBuilder.append(line);
            }
        } catch (IOException e) {
            logger.error(e.getMessage());
        } finally {
            if (bufferedReader != null) {
                try {
                    bufferedReader.close();
                } catch (IOException e) {
                    logger.error(e.getMessage());
                }
            }
        }
        return stringBuilder.toString();
    }
     
    ответ дан Jitender Chahar, с репутацией 166, 29.02.2016