Two structs InputReader
and OutputWriter
for fast and convenient IO in Rust.
The main use of these structs is in competitive programming or Kattis style problems. Reading particularly numbers from stdin via io::stdin()
is not very convenient. In competitive programming you want to be able to easily get the next number or word in the stream since you know the exact format of the input before hand. The InputReader
makes that trivial while also being fast.
Regular output via println!()
can also be problematic since it is line buffered. This can make output can be surprisingly slow. The OutputWriter
buffers all output which results in a much better performance. See documentation below.
To use these in competitive programming simply download the files from here. Then simply put them in the same folder as your solution and import it like below.
This was inspired by this amazing Java class but written completely separately.
```Rust // import it mod inputreader; use inputreader::InputReader;
// ...
fn main() -> std::io::Result<()> { // Create a reader from stdin let mut input = InputReader::new();
// ... or from a file let mut input = InputReader::from_file("input.txt")?;
// ... or from any struct that implements the Read trait. // The reader needs to be wrapped in a Box. // In this example from a tcp stream: let tcpstream = Box::new( TcpStream::connect("127.0.0.1:34254") ); let mut input = InputReader::fromreader(tcp_stream);
// Read numbers and words from the input source simply like this. let x: usize = input.nextusize()?; let y: i64 = input.nexti64()?; let z: f32 = input.nextf32()?; let word: String = input.nextword()?.tostring(); let line: String = input.nextline()?.to_string(); Ok(()) } ```
This struct sacrifices some functionality for performance:
- This does not support UTF8 strings. It will treat each byte in the input source as a separate character. This is a significant speed up and in competitive programming almost always only ascii is used anyway.
- It will not do any validation on the size of numbers before trying to fit them in a u32
for example. This is also fine for competitive programming since number bounds are usually given.
Rust
// Constructs an InputReader which reads from stdin.
InputReader::new() -> InputReader
Rust
// Constructs an InputReader which reads from the file at the given path.
// Note that this returns a result since it will try to open the file.
InputReader::from_file(path: &str) -> Result<InputReader>
Rust
// Constructs an InputReader that reads from the given reader.
InputReader::from_reader(reader: Box<dyn Read>) -> InputReader
The following methods are pretty self-explanatory. They read the next thing from the input source.
```Rust
InputReader::next_usize(&mut self) -> Result
InputReader::nexti8(&mut self) -> Result
InputReader::nextu8(&mut self) -> Result
InputReader::nextf32(&mut self) -> Result
The two string methods return a &str
instead of a String
for optimization reasons.
Rust
InputReader::next_word(&mut self) -> Result<&str>
InputReader::next_line(&mut self) -> Result<&str>
If you need a String
that you own you can convert it by doing input.next_word()?.to_string()
.
```Rust
// Returns Ok(true) if there is more data to be read, Ok(false) otherwise.
InputReader::has_more(&mut self) -> Result
// Changes the internal buffer size. Default: 2^16 bytes // Will return error if shrinking the buffer will cause data loss. InputReader::setbufsize(&mut self, buf_size: usize) -> Result<()> ```
```Rust // import it mod outputwriter; use outputwriter::OutputWriter;
// ...
fn main() -> std::io::Result<()> { // Create a writer from stdout let mut output = OutputWriter::new();
// ... or from a file let mut input = OutputWriter::from_file("input.txt")?;
// ... or from any struct that implements the Write trait. // The writer needs to be wrapped in a Box. // In this example from a tcp stream: let tcpstream = Box::new( TcpStream::connect("127.0.0.1:34254") ); let mut input = OutputWriter::fromwriter(tcp_stream);
// Write to the output source simply like this. output.writeln("Hello world!"); output.write(&format!("{} is a cool number.\n", 1337));
// Optionally you can manually flush the writer. // This will be done automatically when the writer is dropped. output.flush()?; Ok(()) } ```
Rust
// Constructs an OutputWriter which writes to stdout.
OutputWriter::new() -> OutputWriter
Rust
// Constructs an OutputWriter which writes to the file at the given path.
// Note that this returns a result since it will try to open the file.
OutputWriter::from_file(path: &str) -> Result<OutputWriter>
Rust
// Constructs an OutputWriter that writes to the given writer.
OutputWriter::from_reader(reader: Box<dyn Write>) -> OutputWriter
```Rust // Writes the string to the output source. OutputWriter::write(&mut self, s: &str)
// Convenience method for writing the given string with a newline appended. OutputWriter::writeln(&mut self, s: &str)
// Flushes the internal buffer and writes it to the output source. // Note that this is called on drop so you do not have to do it manually. OutputWriter::flush(&mut self) -> Result<()> ```