SL.io
SL.io: Iostream
iostream
s is a type safe, extensible, formatted and unformatted I/O library for streaming I/O.
It supports multiple (and user extensible) buffering strategies and multiple locales.
It can be used for conventional I/O, reading and writing to memory (string streams),
and user-defined extensions, such as streaming across networks (asio: not yet standardized).
Iostream rule summary:
- SL.io.1: Use character-level input only when you have to
- SL.io.2: When reading, always consider ill-formed input
- SL.io.3: Prefer iostreams for I/O
- SL.io.10: Unless you use
printf
-family functions callios_base::sync_with_stdio(false)
- SL.io.50: Avoid
endl
- ???
SL.io.1: Use character-level input only when you have to
Reason
Unless you genuinely just deal with individual characters, using character-level input leads to the user code performing potentially error-prone and potentially inefficient composition of tokens out of characters.
Example
char c;
char buf[128];
int i = 0;
while (cin.get(c) && !isspace(c) && i < 128)
buf[i++] = c;
if (i == 128) {
// ... handle too long string ....
}
Better (much simpler and probably faster):
string s;
s.reserve(128);
cin >> s;
and the reserve(128)
is probably not worthwhile.
Enforcement
???
SL.io.2: When reading, always consider ill-formed input
Reason
Errors are typically best handled as soon as possible. If input isn't validated, every function must be written to cope with bad data (and that is not practical).
Example
???
Enforcement
???
SL.io.3: Prefer iostream
s for I/O
Reason
iostream
s are safe, flexible, and extensible.
Example
// write a complex number:
complex<double> z{ 3, 4 };
cout << z << '\n';
complex
is a user-defined type and its I/O is defined without modifying the iostream
library.
Example
// read a file of complex numbers:
for (complex<double> z; cin >> z; )
v.push_back(z);
Exception
??? performance ???
Discussion: iostream
s vs. the printf()
family
It is often (and often correctly) pointed out that the printf()
family has two advantages compared to iostream
s:
flexibility of formatting and performance.
This has to be weighed against iostream
s advantages of extensibility to handle user-defined types, resilience against security violations,
implicit memory management, and locale
handling.
If you need I/O performance, you can almost always do better than printf()
.
gets()
, scanf()
using %s
, and printf()
using %s
are security hazards (vulnerable to buffer overflow and generally error-prone).
C11 defines some "optional extensions" that do extra checking of their arguments.
If present in your C library, gets_s()
, scanf_s()
, and printf_s()
might be safer alternatives, but they are still not type safe.
Enforcement
Optionally flag <cstdio>
and <stdio.h>
.
SL.io.10: Unless you use printf
-family functions call ios_base::sync_with_stdio(false)
Reason
Synchronizing iostreams
with printf-style
I/O can be costly.
cin
and cout
are by default synchronized with printf
.
Example
int main()
{
ios_base::sync_with_stdio(false);
// ... use iostreams ...
}
Enforcement
???
SL.io.50: Avoid endl
Reason
The endl
manipulator is mostly equivalent to '\n'
and "\n"
;
as most commonly used it simply slows down output by doing redundant flush()
s.
This slowdown can be significant compared to printf
-style output.
Example
cout << "Hello, World!" << endl; // two output operations and a flush
cout << "Hello, World!\n"; // one output operation and no flush
Note
For cin
/cout
(and equivalent) interaction, there is no reason to flush; that's done automatically.
For writing to a file, there is rarely a need to flush
.
Note
For string streams (specifically ostringstream
), the insertion of an endl
is entirely equivalent
to the insertion of a '\n'
character, but also in this case, endl
might be significantly slower.
endl
does not take care of producing a platform specific end-of-line sequence (like "\r\n"
on
Windows). So for a string stream, s << endl
just inserts a single character, '\n'
.
Note
Apart from the (occasionally important) issue of performance,
the choice between '\n'
and endl
is almost completely aesthetic.