Inputstream 轉換 String,你會了嗎?
大家好,我是指北君。
在本教程中,我們將講講如何將一個 InputStream 轉換為一個字符串。
我們將從使用普通的 Java 開始,包括 Java 8+ 的解決方案,然后也會研究使用 Guava 和 Apache Commons IO 庫。
用 Java 進行轉換 - StringBuilder
讓我們看看一個簡單的、低級別的方法,使用普通的 Java,一個 InputStream 和一個簡單的 StringBuilder。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
StringBuilder textBuilder = new StringBuilder();
try (Reader reader = new BufferedReader(new InputStreamReader
(inputStream, Charset.forName(StandardCharsets.UTF_8.name())))) {
int c = 0;
while ((c = reader.read()) != -1) {
textBuilder.append((char) c);
}
}
assertEquals(textBuilder.toString(), originalString);
}
用 Java 8 進行轉換 -- BufferedReader
Java 8 給 BufferedReader 帶來了一個新的 lines() 方法。讓我們看看如何利用它將一個 InputStream 轉換為一個字符串。
@Test
public void convertingAnInputStreamToAString() {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
String text = new BufferedReader(
new InputStreamReader(inputStream, StandardCharsets.UTF_8))
.lines()
.collect(Collectors.joining("\n"));
assertThat(text, equalTo(originalString));
}
值得一提的是,lines() 使用的是 readLine() 方法。readLine() 假定一行是由換行("\n")、回車("\r")或回車后立即換行中的任何一種結束符。換句話說,它支持所有常見的行結束方式。
另一方面,當我們使用 Collectors.join() 時,我們需要明確決定我們要為創建的 String 使用哪種類型的結束符。
我們也可以使用 Collectors.join(System.lineSeparator()) ,在這種情況下,輸出結果取決于系統設置。
用 Java 9+ 進行轉換 - InputStream.readAllBytes()
如果我們在 Java 9 或以上版本,我們可以利用一個新的 readAllBytes 方法添加到 InputStream 中。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
String text = new String(inputStream.readAllBytes(), StandardCharsets.UTF_8);
assertThat(text, equalTo(originalString));
}
我們需要注意的是,這段簡單的代碼是為那些方便將所有字節讀入字節數組的簡單情況準備的。我們不應該用它來讀取有大量數據的輸入流。
用 Java Scanner 進行轉換
接下來,讓我們看看一個使用標準文本掃描器的普通Java例子。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
String text = null;
try (Scanner scanner = new Scanner(inputStream, StandardCharsets.UTF_8.name())) {
text = scanner.useDelimiter("\\A").next();
}
assertThat(text, equalTo(originalString));
}
請注意,InputStream 將被關閉的 Scanner 關閉。
同樣值得澄清的是 useDelimiter("\A") 的作用。這里我們傳遞了'\A',它是一個邊界標記重碼,表示輸入的開始。本質上,這意味著 next() 調用讀取了整個輸入流。
使用 ByteArrayOutputStream 進行轉換
最后,讓我們看看另一個普通的Java例子,這次是使用 ByteArrayOutputStream 類。
@Test
public void convertingAnInputStreamToString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
int nRead;
byte[] data = new byte[1024];
while ((nRead = inputStream.read(data, 0, data.length)) != -1) {
buffer.write(data, 0, nRead);
}
buffer.flush();
byte[] byteArray = buffer.toByteArray();
String text = new String(byteArray, StandardCharsets.UTF_8);
assertThat(text, equalTo(originalString));
}
在這個例子中,InputStream 通過讀寫字節塊被轉換為 ByteArrayOutputStream。然后 OutputStream 被轉換為一個字節數組,用來創建一個字符串。
用java.nio進行轉換
另一個解決方案是將 InputStream 的內容復制到一個文件中,然后將其轉換為一個字符串。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
Path tempFile =
Files.createTempDirectory("").resolve(UUID.randomUUID().toString() + ".tmp");
Files.copy(inputStream, tempFile, StandardCopyOption.REPLACE_EXISTING);
String result = new String(Files.readAllBytes(tempFile));
assertThat(result, equalTo(originalString));
}
這里我們使用 java.nio.file.Files 類來創建一個臨時文件,同時將 InputStream 的內容復制到文件中。然后用同一個類用 readAllBytes() 方法將文件內容轉換為一個字符串。
用Guava進行轉換
讓我們從一個利用 ByteSource 功能的 Guava 例子開始。
@Test
public void convertingAnInputStreamToAString()
throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
ByteSource byteSource = new ByteSource() {
@Override
public InputStream openStream() throws IOException {
return inputStream;
}
};
String text = byteSource.asCharSource(Charsets.UTF_8).read();
assertThat(text, equalTo(originalString));
}
讓我們來看看這些步驟:
首先,我們把我們的 InputStream 包裝成一個 ByteSource.
其次,我們把 ByteSource 看作是一個具有 UTF8 字符集的 CharSource。
最后,我們使用 CharSource 將其作為一個字符串來讀取。
一個更簡單的轉換方法是使用 Guava,但需要明確地關閉流, 我們可以簡單地使用 try-with-resources 語法來處理這個問題。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
String text = null;
try (Reader reader = new InputStreamReader(inputStream)) {
text = CharStreams.toString(reader);
}
assertThat(text, equalTo(originalString));
}
用 Apache Commons IO 進行轉換
現在讓我們來看看如何用 Commons IO 庫來做這個。
一個重要的注意事項是,與 Guava 不同的是,這些例子都不會關閉 InputStream。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
String text = IOUtils.toString(inputStream, StandardCharsets.UTF_8.name());
assertThat(text, equalTo(originalString));
}
我們也可以用一個StringWriter來做轉換。
ream 轉換為一個字符串。我們從使用普通 Java 開始,然后探索了如何使用 Guava 和 Apache Commons IO 庫。
@Test
public void convertingAnInputStreamToAString() throws IOException {
String originalString = randomString(8);
InputStream inputStream = new ByteArrayInputStream(originalString.getBytes());
StringWriter writer = new StringWriter();
String encoding = StandardCharsets.UTF_8.name();
IOUtils.copy(inputStream, writer, encoding);
assertThat(writer.toString(), equalTo(originalString));
}