it-swarm.asia

كيف يمكنني قراءة/تحويل InputStream إلى سلسلة في Java؟

إذا كان لديك كائن Java.io.InputStream ، فكيف يجب عليك معالجة هذا الكائن وإنتاج String؟


افترض أن لدي InputStream يحتوي على بيانات نصية ، وأريد تحويله إلى String ، على سبيل المثال يمكنني كتابة ذلك في ملف سجل.

ما هي أسهل طريقة لأخذ InputStream وتحويله إلى String؟

public String convertStreamToString(InputStream is) {
    // ???
}
3650
Johnny Maelstrom

هناك طريقة لطيفة للقيام بذلك باستخدام commons ApacheIOUtils لنسخ 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 إذا كنت لا ترغب في مزج التدفقات والكتاب

2292
Harry Lime

في ما يلي طريقة لاستخدام مكتبة Java القياسية فقط (لاحظ أن الدفق غير مغلق ، قد يختلف عدد الأميال المقطوعة).

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

لقد تعلمت هذه الخدعة من "حيل الماسح الغبي" مقال. السبب في أنه يعمل بسبب الماسح الضوئي يتكرر عبر الرموز المميزة في الدفق ، وفي هذه الحالة نفصل الرموز المميزة باستخدام "بداية حدود الإدخال" (\ A) ، وبالتالي نقدم لنا رمز واحد فقط للمحتويات الكاملة لل مجرى.

ملاحظة ، إذا كنت بحاجة إلى أن تكون محددًا حول ترميز دفق الإدخال ، فيمكنك تقديم الوسيطة الثانية إلى مُنشئ Scanner الذي يشير إلى مجموعة الأحرف المراد استخدامها (مثل "UTF-8").

غيض قبعة يذهب أيضا إلى يعقوب ، الذي أشار لي مرة واحدة إلى المادة المذكورة.

2209
Pavel Repin

لخص الإجابات الأخرى التي وجدت 11 طريقة رئيسية للقيام بذلك (انظر أدناه). وقد كتبت بعض اختبارات الأداء (انظر النتائج أدناه):

طرق تحويل InputStream إلى سلسلة:

  1. باستخدام IOUtils.toString (يستخدم أباتشي)

    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. استخدام دفق API (جافا 8). تحذير : هذا الحل يحول فواصل الأسطر المختلفة (مثل \r\n) إلى \n.

    String result = new BufferedReader(new InputStreamReader(inputStream))
      .lines().collect(Collectors.joining("\n"));
    
  5. استخدام واجهة دفق متوازية (جافا 8). تحذير : هذا الحل يحول فواصل الأسطر المختلفة (مثل \r\n) إلى \n.

    String result = new BufferedReader(new InputStreamReader(inputStream)).lines()
       .parallel().collect(Collectors.joining("\n"));
    
  6. باستخدام InputStreamReader و StringBuilder (JDK)

    final int bufferSize = 1024;
    final char[] buffer = new char[bufferSize];
    final StringBuilder out = new StringBuilder();
    Reader in = new InputStreamReader(inputStream, "UTF-8");
    for (; ; ) {
        int rsz = in.read(buffer, 0, buffer.length);
        if (rsz < 0)
            break;
        out.append(buffer, 0, rsz);
    }
    return out.toString();
    
  7. باستخدام StringWriter و IOUtils.copy (Apache Commons)

    StringWriter writer = new StringWriter();
    IOUtils.copy(inputStream, writer, "UTF-8");
    return writer.toString();
    
  8. باستخدام ByteArrayOutputStream و inputStream.read (JDK)

    ByteArrayOutputStream result = new ByteArrayOutputStream();
    byte[] buffer = new byte[1024];
    int length;
    while ((length = inputStream.read(buffer)) != -1) {
        result.write(buffer, 0, length);
    }
    // StandardCharsets.UTF_8.name() > JDK 7
    return result.toString("UTF-8");
    
  9. باستخدام BufferedReader (JDK). تحذير: هذا الحل يحول فواصل الأسطر المختلفة (مثل \n\r) إلى خاصية نظام line.separator (على سبيل المثال ، في Windows إلى "\ r\n").

    String newLine = System.getProperty("line.separator");
    BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
    StringBuilder result = new StringBuilder();
    boolean flag = false;
    for (String line; (line = reader.readLine()) != null; ) {
        result.append(flag? newLine: "").append(line);
        flag = true;
    }
    return result.toString();
    
  10. باستخدام BufferedInputStream و ByteArrayOutputStream (JDK)

    BufferedInputStream bis = new BufferedInputStream(inputStream);
    ByteArrayOutputStream buf = new ByteArrayOutputStream();
    int result = bis.read();
    while(result != -1) {
        buf.write((byte) result);
        result = bis.read();
    }
    // StandardCharsets.UTF_8.name() > JDK 7
    return buf.toString("UTF-8");
    
  11. باستخدام inputStream.read() و StringBuilder (JDK). تحذير : هذا الحل لديه مشاكل مع Unicode ، على سبيل المثال مع النص الروسي (يعمل بشكل صحيح فقط مع نص غير Unicode)

    int ch;
    StringBuilder sb = new StringBuilder();
    while((ch = inputStream.read()) != -1)
        sb.append((char)ch);
    reset();
    return sb.toString();
    

تحذير :

  1. الحلول 4 و 5 و 9 تحويل فواصل الأسطر المختلفة إلى واحد.

  2. لا يمكن أن يعمل الحل 11 بشكل صحيح مع نص Unicode

اختبارات الأداء

اختبارات الأداء لـ String (الطول = 175) ، url in github (mode = Average Time ، system = Linux ، علامة 1،343 هي الأفضل):

              Benchmark                         Mode  Cnt   Score   Error  Units
 8. ByteArrayOutputStream and read (JDK)        avgt   10   1,343 ± 0,028  us/op
 6. InputStreamReader and StringBuilder (JDK)   avgt   10   6,980 ± 0,404  us/op
10. BufferedInputStream, ByteArrayOutputStream  avgt   10   7,437 ± 0,735  us/op
11. InputStream.read() and StringBuilder (JDK)  avgt   10   8,977 ± 0,328  us/op
 7. StringWriter and IOUtils.copy (Apache)      avgt   10  10,613 ± 0,599  us/op
 1. IOUtils.toString (Apache Utils)             avgt   10  10,605 ± 0,527  us/op
 3. Scanner (JDK)                               avgt   10  12,083 ± 0,293  us/op
 2. CharStreams (guava)                         avgt   10  12,999 ± 0,514  us/op
 4. Stream Api (Java 8)                         avgt   10  15,811 ± 0,605  us/op
 9. BufferedReader (JDK)                        avgt   10  16,038 ± 0,711  us/op
 5. parallel Stream Api (Java 8)                avgt   10  21,544 ± 0,583  us/op

اختبارات الأداء لـ String (الطول = 50100) ، عنوان url في github (الوضع = متوسط ​​الوقت ، النظام = Linux ، النقاط 200،715 هي الأفضل):

               Benchmark                        Mode  Cnt   Score        Error  Units
 8. ByteArrayOutputStream and read (JDK)        avgt   10   200,715 ±   18,103  us/op
 1. IOUtils.toString (Apache Utils)             avgt   10   300,019 ±    8,751  us/op
 6. InputStreamReader and StringBuilder (JDK)   avgt   10   347,616 ±  130,348  us/op
 7. StringWriter and IOUtils.copy (Apache)      avgt   10   352,791 ±  105,337  us/op
 2. CharStreams (guava)                         avgt   10   420,137 ±   59,877  us/op
 9. BufferedReader (JDK)                        avgt   10   632,028 ±   17,002  us/op
 5. parallel Stream Api (Java 8)                avgt   10   662,999 ±   46,199  us/op
 4. Stream Api (Java 8)                         avgt   10   701,269 ±   82,296  us/op
10. BufferedInputStream, ByteArrayOutputStream  avgt   10   740,837 ±    5,613  us/op
 3. Scanner (JDK)                               avgt   10   751,417 ±   62,026  us/op
11. InputStream.read() and StringBuilder (JDK)  avgt   10  2919,350 ± 1101,942  us/op

الرسوم البيانية (اختبارات الأداء حسب طول مدخلات البث في نظام ويندوز 7)
 enter image description here

اختبار الأداء (متوسط ​​الوقت) حسب طول دفق الإدخال في نظام Windows 7:

 length  182    546     1092    3276    9828    29484   58968

 test8  0.38    0.938   1.868   4.448   13.412  36.459  72.708
 test4  2.362   3.609   5.573   12.769  40.74   81.415  159.864
 test5  3.881   5.075   6.904   14.123  50.258  129.937 166.162
 test9  2.237   3.493   5.422   11.977  45.98   89.336  177.39
 test6  1.261   2.12    4.38    10.698  31.821  86.106  186.636
 test7  1.601   2.391   3.646   8.367   38.196  110.221 211.016
 test1  1.529   2.381   3.527   8.411   40.551  105.16  212.573
 test3  3.035   3.934   8.606   20.858  61.571  118.744 235.428
 test2  3.136   6.238   10.508  33.48   43.532  118.044 239.481
 test10 1.593   4.736   7.527   20.557  59.856  162.907 323.147
 test11 3.913   11.506  23.26   68.644  207.591 600.444 1211.545
2051
Viacheslav Vedenin

يسمح أباتشي كومنز:

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

بالطبع ، يمكنك اختيار ترميزات الأحرف الأخرى إلى جانب UTF-8.

انظر أيضا: ( وثائق )

823
Chinnery

مع الأخذ في الاعتبار الملف ، يجب أولاً الحصول على مثيل 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();
}
281
Paul de Vrieze

استعمال:

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();
238
sampathpremarathna

إذا كنت تستخدم Google-Collections/Guava ، فيمكنك القيام بما يلي:

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

لاحظ أن المعلمة الثانية (على سبيل المثال Charsets.UTF_8) الخاصة بـ InputStreamReader ليست ضرورية ، ولكن من الأفضل عمومًا تحديد الترميز إذا كنت تعرف ذلك (وهو ما يجب عليك!)

165
Sakuraba

هذا هو حل 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;
}
113
TacB0sS

إليك حل 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();
}
59
Drew Noakes

استعمال:

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();
}
58
Jon Moore

للإكمال هنا هو Java 9 solution:

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

readAllBytes موجود حاليًا في قاعدة بيانات JDK 9 الرئيسية ، لذلك من المحتمل أن يظهر في الإصدار. يمكنك تجربة ذلك الآن باستخدام JDK 9 builds لقطة .

51
Tagir Valeev

فعلت معيارًا على 14 إجابة مميزة هنا (آسف لعدم تقديم ائتمانات ولكن هناك الكثير من التكرارات).

والنتيجة مفاجئة للغاية. اتضح أن أباتشي 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
  • نيوستريم: 198
  • Java9ISTransferTo: 201
  • Java9ISReadAllBytes: 205
  • BufferedInputStreamVsByteArrayOutputStream: 314
  • ApacheStringWriter2: 574
  • جوافاشار ستريم: 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);
        }
    }

}
44
Ilya Gazman

أود استخدام بعض الحيل جافا 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.
    }
}

أساسا نفس بعض الإجابات الأخرى باستثناء أكثر إيجازا.

37
Simon Kuang

أجريت بعض اختبارات التوقيت لأن الوقت مهم دائمًا.

حاولت الحصول على الرد في سلسلة 3 طرق مختلفة. (ظاهر أدناه)
تركت كتل المحاولة/الصيد من أجل سهولة القراءة.

لإعطاء السياق ، هذا هو الكود السابق لكل الطرق الثلاثة:

   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

أي من هذه الطرق هو الحل المناسب للحصول على استجابة وإنشاء سلسلة منه.

30
Brett Holt

حل 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()));
    }
}

كما ذكر Christoffer Hammarström أدناه إجابة أخرى من الآمن تحديد Charset بشكل صريح. أي. يمكن أن يكون مُنشئ InputStreamReader التغييرات كما يلي:

new InputStreamReader(is, Charset.forName("UTF-8"))
28
czerny

فيما يلي إجابة 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();
}
24
TKH

إذا كنت تشعر بالمغامرة ، يمكنك مزج سكالا وجافا وينتهي بهذا:

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

إن مزج كود Java و Scala والمكتبات له فوائد.

انظر الوصف الكامل هنا: الطريقة المثالية لتحويل InputStream إلى سلسلة في Scala

22
Jack

إذا لم تتمكن من استخدام العموم 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);
    }
}
19
DJDaveMark

هذه إجابة مقتبسة من org.Apache.commons.io.IOUtilssource code ، لأولئك الذين يريدون تطبيق Apache لكنهم لا يريدون المكتبة بأكملها.

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();
}
18
Dreaming in Code

تأكد من إغلاق التدفقات في النهاية إذا كنت تستخدم 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();
}

تحرير: في 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();
    }
}
16
Thamme Gowda

فيما يلي الطريقة الكاملة لتحويل 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();
}
15
laksys

إليك كيفية القيام بذلك باستخدام JDK فقط باستخدام مخازن صفيف البايت. هذه هي الطريقة التي تعمل بها جميع المشاع- 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);
14
Matt Shannon

واحدة أخرى ، لجميع مستخدمي الربيع:

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 ، لكنها تترك الدفق مفتوحًا عند الانتهاء.

12
James

استخدم Java.io.InputStream.transferTo (OutputStream) المدعومة في Java 9 و ByteArrayOutputStream.toString (سلسلة) التي تأخذ اسم مجموعة الأحرف:

public static String gobble(InputStream in, String charsetName) throws IOException {
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    in.transferTo(bos);
    return bos.toString(charsetName);
}
12
jmehrens

مستخدمو Kotlin يفعلون ببساطة:

println(InputStreamReader(is).readText())

بينما

readText()

هي طريقة التمديد المدمجة في مكتبة Kotlin.

12
Alex

هذا جميل لأن:

  • سلامة اليد charset.
  • يمكنك التحكم في حجم المخزن المؤقت للقراءة.
  • يمكنك توفير طول المنشئ ويمكن أن لا يكون بالضبط.
  • خالية من التبعيات المكتبة.
  • هو لجافا 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();
}
9
Daniel De León

إليك حل Java 8 المستند إلى الحل ، والذي يستخدم واجهة برمجة تطبيقات Stream جديدة لجمع كل الأسطر من InputStream:

public static String toString(InputStream inputStream) {
    BufferedReader reader = new BufferedReader(
        new InputStreamReader(inputStream));
    return reader.lines().collect(Collectors.joining(
        System.getProperty("line.separator")));
}
6
Christian Rädel

في رائع

inputStream.getText()
5
Snekse

فيما يتعلق بـ reduce و concat ، يمكن التعبير عنها في Java 8 على النحو التالي:

String fromFile = new BufferedReader(new   
InputStreamReader(inputStream)).lines().reduce(String::concat).get();
5
libnull-dev

لقد كتبت فصلاً يفعل ذلك تمامًا ، لذلك اعتقدت أنني سأشاركه مع الجميع. في بعض الأحيان لا ترغب في إضافة 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);
        }
    }
}
4
Ben Barkay

استنادًا إلى الجزء الثاني من إجابة Apache Commons المقبولة ولكن مع ملء الفجوة الصغيرة لإغلاق الدفق دائمًا:

    String theString;
    try {
        theString = IOUtils.toString(inputStream, encoding);
    } finally {
        IOUtils.closeQuietly(inputStream);
    }
4
Steve Chambers
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);
4
Vaishali Sutariya

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 Scanner؟

4
Halfacht

حسنًا ، يمكنك برمجة البرنامج بنفسك ... إنه ليس معقدًا ...

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();
    }
4
Victor

عملت أدناه رمز بالنسبة لي.

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

لست متأكدًا مما إذا كان هناك أي مشكلات ترميز. يرجى التعليق ، إذا كان سيكون هناك أي مشاكل مع رمز.

4
Anand N

توفر 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"));
4
Vadzim
public String read(InputStream in) throws IOException {
    try (BufferedReader buffer = new BufferedReader(new InputStreamReader(in))) {
        return buffer.lines().collect(Collectors.joining("\n"));
    }
}
3
Hao Zheng

مع Okio:

String result = Okio.buffer(Okio.source(inputStream)).readUtf8();
3
drakeet

الإجابة 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));
  }
}
3
Brian Pontarelli

يمكنك استخدام اباتشي كومنز.

في 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));
}
3
Rys

ISO-8859-1

إليك طريقة أداء {جدا لفعل ذلك إذا كنت تعرف أن ترميز تدفق الإدخال الخاص بك هو ISO-8859-1 أو ASCII. إنه (1) يتجنب المزامنة غير الضرورية الموجودة في StringWriter's StringBuffer ، (2) يتجنب الحمل الزائد لـ InputStreamReader ، و (3) يقلل من عدد المرات StringBuilder's char الصفيف الداخلي.

public static String iso_8859_1(InputStream is) throws IOException {
    StringBuilder chars = new StringBuilder(Math.max(is.available(), 4096));
    byte[] buffer = new byte[4096];
    int n;
    while ((n = is.read(buffer)) != -1) {
        for (int i = 0; i < n; i++) {
            chars.append((char)(buffer[i] & 0xFF));
        }
    }
    return chars.toString();
}

UTF-8

يمكن استخدام نفس الاستراتيجية العامة لتيار مشفر باستخدام UTF-8:

public static String utf8(InputStream is) throws IOException {
    StringBuilder chars = new StringBuilder(Math.max(is.available(), 4096));
    byte[] buffer = new byte[4096];
    int n;
    int state = 0;
    while ((n = is.read(buffer)) != -1) {
        for (int i = 0; i < n; i++) {
            if ((state = nextStateUtf8(state, buffer[i])) >= 0) {
                chars.appendCodePoint(state);
            } else if (state == -1) { //error
                state = 0;
                chars.append('\uFFFD'); //replacement char
            }
        }
    }
    return chars.toString();
}

حيث يتم تعريف الدالة nextStateUtf8() كما يلي:

/**
 * Returns the next UTF-8 state given the next byte of input and the current state.
 * If the input byte is the last byte in a valid UTF-8 byte sequence,
 * the returned state will be the corresponding unicode character (in the range of 0 through 0x10FFFF).
 * Otherwise, a negative integer is returned. A state of -1 is returned whenever an
 * invalid UTF-8 byte sequence is detected.
 */
static int nextStateUtf8(int currentState, byte nextByte) {
    switch (currentState & 0xF0000000) {
        case 0:
            if ((nextByte & 0x80) == 0) { //0 trailing bytes (ASCII)
                return nextByte;
            } else if ((nextByte & 0xE0) == 0xC0) { //1 trailing byte
                if (nextByte == (byte) 0xC0 || nextByte == (byte) 0xC1) { //0xCO & 0xC1 are overlong
                    return -1;
                } else {
                    return nextByte & 0xC000001F;
                }
            } else if ((nextByte & 0xF0) == 0xE0) { //2 trailing bytes
                if (nextByte == (byte) 0xE0) { //possibly overlong
                    return nextByte & 0xA000000F;
                } else if (nextByte == (byte) 0xED) { //possibly surrogate
                    return nextByte & 0xB000000F;
                } else {
                    return nextByte & 0x9000000F;
                }
            } else if ((nextByte & 0xFC) == 0xF0) { //3 trailing bytes
                if (nextByte == (byte) 0xF0) { //possibly overlong
                    return nextByte & 0x80000007;
                } else {
                    return nextByte & 0xE0000007;
                }
            } else if (nextByte == (byte) 0xF4) { //3 trailing bytes, possibly undefined
                return nextByte & 0xD0000007;
            } else {
                return -1;
            }
        case 0xE0000000: //3rd-to-last continuation byte
            return (nextByte & 0xC0) == 0x80 ? currentState << 6 | nextByte & 0x9000003F : -1;
        case 0x80000000: //3rd-to-last continuation byte, check overlong
            return (nextByte & 0xE0) == 0xA0 || (nextByte & 0xF0) == 0x90 ? currentState << 6 | nextByte & 0x9000003F : -1;
        case 0xD0000000: //3rd-to-last continuation byte, check undefined
            return (nextByte & 0xF0) == 0x80 ? currentState << 6 | nextByte & 0x9000003F : -1;
        case 0x90000000: //2nd-to-last continuation byte
            return (nextByte & 0xC0) == 0x80 ? currentState << 6 | nextByte & 0xC000003F : -1;
        case 0xA0000000: //2nd-to-last continuation byte, check overlong
            return (nextByte & 0xE0) == 0xA0 ? currentState << 6 | nextByte & 0xC000003F : -1;
        case 0xB0000000: //2nd-to-last continuation byte, check surrogate
            return (nextByte & 0xE0) == 0x80 ? currentState << 6 | nextByte & 0xC000003F : -1;
        case 0xC0000000: //last continuation byte
            return (nextByte & 0xC0) == 0x80 ? currentState << 6 | nextByte & 0x3F : -1;
        default:
            return -1;
    }
}

الكشف التلقائي الترميز

إذا تم تشفير دفق الإدخال باستخدام ASCII أو ISO-8859-1 أو UTF-8 ، لكنك لست متأكدًا من ذلك ، يمكننا استخدام طريقة مشابهة للطريقة الأخيرة ، ولكن مع مكون إضافي للكشف عن التشفير إلى الكشف التلقائي الترميز قبل إرجاع السلسلة.

public static String autoDetect(InputStream is) throws IOException {
    StringBuilder chars = new StringBuilder(Math.max(is.available(), 4096));
    byte[] buffer = new byte[4096];
    int n;
    int state = 0;
    boolean ascii = true;
    while ((n = is.read(buffer)) != -1) {
        for (int i = 0; i < n; i++) {
            if ((state = nextStateUtf8(state, buffer[i])) > 0x7F)
                ascii = false;
            chars.append((char)(buffer[i] & 0xFF));
        }
    }

    if (ascii || state < 0) { //probably not UTF-8
        return chars.toString();
    }
    //probably UTF-8
    int pos = 0;
    char[] charBuf = new char[2];
    for (int i = 0, len = chars.length(); i < len; i++) {
        if ((state = nextStateUtf8(state, (byte)chars.charAt(i))) >= 0) {
            boolean hi = Character.toChars(state, charBuf, 0) == 2;
            chars.setCharAt(pos++, charBuf[0]);
            if (hi) {
                chars.setCharAt(pos++, charBuf[1]);
            }
        }
    }
    return chars.substring(0, pos);
}

إذا كان دفق الإدخال الخاص بك يحتوي على ترميز ليس ISO-8859-1 ولا ASCII ولا UTF-8 ، فأنا أتأخر عن الإجابات الأخرى الموجودة بالفعل.

2
Hans Brende

ما يلي لا يعالج السؤال الأصلي ، بل بعض الإجابات.

العديد من الردود تشير إلى حلقات من النموذج

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

أو

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

يلوث النموذج الأول مساحة الاسم للنطاق المحيط بإعلان متغير "قراءة" في النطاق المغلق والذي لن يتم استخدامه لأي شيء خارج الحلقة for. يكرر النموذج الثاني استدعاء readline ().

إليك طريقة أكثر نظافة لكتابة هذا النوع من الحلقة في Java. اتضح أن الجملة الأولى في حلقة for-loop لا تتطلب قيمة مهيئ فعلية. هذا يحافظ على نطاق "الخط" المتغير داخل نص الحلقة for. أكثر أناقة بكثير! لم أر أي شخص يستخدم هذا النموذج في أي مكان (لقد اكتشفته بشكل عشوائي قبل يوم واحد) ، لكنني استخدمه طوال الوقت.

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

يمكنك استخدام الصبار :

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

ترميز UTF-8 هو الافتراضي. إذا كنت بحاجة إلى واحدة أخرى:

String text = new TextOf(inputStream, "UTF-16").asString();
2
yegor256
InputStreamReader i = new InputStreamReader(s);
BufferedReader str = new BufferedReader(i);
String msg = str.readLine();
System.out.println(msg);

إليك كائن InputStream الذي سيتم تحويله إلى String

2
Omkar Khot

لقد قمت بإنشاء هذا الرمز ، وهو يعمل. لا توجد مكونات خارجية مطلوبة.

هناك تحويل String إلى Stream و Stream إلى String ...

import Java.io.ByteArrayInputStream;
import Java.io.InputStream;

public class STRINGTOSTREAM {

    public static void main(String[] args)
    {
        String text = "Hello Bhola..!\nMy Name Is Kishan ";

        InputStream strm = new ByteArrayInputStream(text.getBytes());    // Convert String to Stream

        String data = streamTostring(strm);

        System.out.println(data);
    }

    static String streamTostring(InputStream stream)
    {
        String data = "";

        try
        {
            StringBuilder stringbuld = new StringBuilder();
            int i;
            while ((i=stream.read())!=-1)
            {
                stringbuld.append((char)i);
            }
            data = stringbuld.toString();
        }
        catch(Exception e)
        {
            data = "No data Streamed.";
        }
        return data;
    }
1
13hola

استعمال:

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

ملاحظة: ربما هذه ليست فكرة جيدة. تستخدم هذه الطريقة التكرار وبالتالي ستصل إلى StackOverflowError بسرعة كبيرة:

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

من فضلك لا تخفض هذا لمجرد أنه خيار سيء للاستخدام ؛ كان هذا في الغالب الإبداعي :)

1
HyperNeutrino

هذا الحل لهذا السؤال ليس هو الأسهل ، ولكن نظرًا لعدم ذكر تدفقات وقنوات 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);
}

يجب أن يكون أداء هذه الطريقة جيدًا للملفات الكبيرة.

1
gil.fernandes
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();
            }
1
Harsh

طريقة لتحويل المدخلات إلى سلسلة

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();
}
1
Jitender Chahar

تم العثور على هذا المقتطف في\sdk\samples\Android-19\connectivity\NetworkConnect\NetworkConnectSample\src\main\Java\com\example\Android\networkconnect\MainActivity.Java المرخص لها بموجب ترخيص Apache ، الإصدار 2.0 وكتبته Google .

/** Reads an InputStream and converts it to a String.
 * @param stream InputStream containing HTML from targeted site.
 * @param len Length of string that this method returns.
 * @return String concatenated according to len parameter.
 * @throws Java.io.IOException
 * @throws Java.io.UnsupportedEncodingException
 */
private String readIt(InputStream stream, int len) throws IOException, UnsupportedEncodingException {
    Reader reader = null;
    reader = new InputStreamReader(stream, "UTF-8");
    char[] buffer = new char[len];
    reader.read(buffer);
    return new String(buffer);
}
1
Fred

String resultString = IOUtils.toString(userInputStream, "UTF-8");

0
Akash

أقترح StringWriter فئة لهذه المشكلة.

StringWriter wt= new StringWriter();
IOUtils.copy(inputStream, wt, encoding);
String st= wt.toString();
0
Hasee Amarathunga

كما يمكنك الحصول على InputStream من مسار مورد محدد:

public static InputStream getResourceAsStream(String path)
{
    InputStream myiInputStream = ClassName.class.getResourceAsStream(path);
    if (null == myiInputStream)
    {
        mylogger.info("Can't find path = ", path);
    }

    return myiInputStream;
}

للحصول على InputStream من مسار معين:

public static URL getResource(String path)
{
    URL myURL = ClassName.class.getResource(path);
    if (null == myURL)
    {
        mylogger.info("Can't find resource path = ", path);
    }
    return myURL;
}
0
Ravi