I am using a BufferedReader
constructor to make a new copy of an existing BufferedReader
.
BufferedReader buffReader = new BufferedReader(originalBuffReader);
The new buffReader
is working fine, but when I do originalBuffReader.readLine()
it gives me null
. Is there any other way I can make a new bufferReader
without affecting my original BufferedReader
.
FYI: I am getting bufferReader as an input to my method; and I do not have a access to the source.
The BufferedReader can't read the InputStream directly; So, we need to use an adapter like InputStreamReader to convert bytes to characters format. For example: // BufferedReader -> InputStreamReader -> InputStream BufferedReader br = new BufferedReader( new InputStreamReader(inputStream, StandardCharsets. UTF_8));
parallel() . This way your input will be split into chunks and further stream operations will be performed on chunks in parallel.
Closes this stream and releases any system resources associated with it. If the stream is already closed then invoking this method has no effect. So, if you don't close(), system resources may be still associated with the reader which may cause memory leak.
Any other way I can make a new bufferReader without affecting my oroginal BufferReader
There's no straight forward way of solving it by just creating two BufferedReader
s. (The two readers will consume data from the same source.) You'll have to add another level of buffering on the source, so each reader can read the stream independently.
This can be achieved by combining TeeInputStream
from Apache Commons and a PipedInputStream
and PipedOutputStream
as follows:
import java.io.*;
import org.apache.commons.io.input.TeeInputStream;
class Test {
public static void main(String[] args) throws IOException {
// Create the source input stream.
InputStream is = new FileInputStream("filename.txt");
// Create a piped input stream for one of the readers.
PipedInputStream in = new PipedInputStream();
// Create a tee-splitter for the other reader.
TeeInputStream tee = new TeeInputStream(is, new PipedOutputStream(in));
// Create the two buffered readers.
BufferedReader br1 = new BufferedReader(new InputStreamReader(tee));
BufferedReader br2 = new BufferedReader(new InputStreamReader(in));
// Do some interleaved reads from them.
System.out.println("One line from br1:");
System.out.println(br1.readLine());
System.out.println();
System.out.println("Two lines from br2:");
System.out.println(br2.readLine());
System.out.println(br2.readLine());
System.out.println();
System.out.println("One line from br1:");
System.out.println(br1.readLine());
System.out.println();
}
}
Output:
One line from br1:
Line1: Lorem ipsum dolor sit amet, <-- reading from start
Two lines from br2:
Line1: Lorem ipsum dolor sit amet, <-- reading from start
Line2: consectetur adipisicing elit,
One line from br1:
Line2: consectetur adipisicing elit, <-- resumes on line 2
Below is one way to consider creating a new BufferedReader out of the Original BufferedReader.
Essentially, we are dumping the contents of the original buffered reader into a string and then recreating a new BufferedReader object. To be able to re read the original BufferedReader a second time you can mark it and then after reading it, you can then reset it.
One thing to keep in mind is you might want to protect by DDOS attacks, where a very large buffered reader can come in and you might want to avoid reading for ever and filling up the memory, you can basically break the for loop after some point or when a certain condition is met.
final String originalBufferedReaderDump;
originalBufferedReaderDump.mark(Integer.MAX_VALUE);
for (String line; (line = originalBufferedReader.readLine()) != null; originalBufferedReaderDump+= line);
originalBufferedReader.reset();
final BufferedReader copiedBufferedReader = new BufferedReader(new InputStreamReader(new ByteArrayInputStream(originalBufferedReaderDump.getBytes())));
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With