01 - intro
Theory
A stream is an abstraction over I/O (input/output) operations. The core goal is to hide implementation-specific details and offer a common interface for the same operations, regardless what the stream is actually connected to. The stream can represent a file, network traffic, in-memory operations such as compression and more - basically any abstract task that involves reading or writing data. Streams can work with any data (they almost always work on byte level) but in majority of cases it is text.
A good example is Linux kernel interface, where for userspace streams are represented with file descriptors (unique integer numbers). File descriptors do not necessarily represent files on disk (they represent anything that supports I/O operations) - in unix philosophy everything is either a file or a process.
Opening a stream looks differently depending on what it is:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
#include <fcntl.h> // open a file - returns positive integer (fd) or -1 in case of error int open(const char* path, int flags, mode_t mode); // create a file - returns positive integer (fd) or -1 in case of error // (yes, this function is named "creat", not "create") int creat(const char* path, mode_t mode); #include <sys/socket.h> // open a socket - returns positive integer (fd) or -1 in case of error int socket(int domain, int type, int protocol); |
but subsequent operations use the same interface:
1 2 3 4 5 6 7 |
#include <unistd.h> // copy up to count bytes from the stream identified by fd to the buffer ssize_t read(int fd, void* buf, size_t count); // copy up to count bytes from buffer to the stream identified by fd ssize_t write(int fd, const void* buf, size_t count); |
Closing streams is also done through the same function:
1 2 3 4 |
#include <unistd.h> // close the stream and release associated resources int close(int fd); |
The beauty of this approach is that once a thing has been set up (socket opened, file created, etc.) the same code can be used to operate on it, regardless of what exactly it represents. This is known as polymorphism and is one of key aspects of OOP (object-oriented programming, explained in later chapters). The respective OOP terms for opening, using and closing objects are construction, methods and destruction.
Even though majority of Linux kernel code is C (which has no explicit features for object-oriented programming), specific patterns from OOP can be seen in its interfaces. Later in the tutorial you will learn dedicated C++ features for OOP.
Standard library streams
C++ streams are a higher level of abstraction - you don't want to directly call OS-level functions when performing I/O - doing so is bug-prone and limits code portability. On platforms where Linux is the kernel, C++ standard library streams are just wrapper code around it that add buffering, error handling, formatting and other features. On other platforms (e.g. Windows) the logic is more complex as the interface mandated by the C++ standard differs from the interface provided by the OS.
Streams in the standard library form a hierarchy of types:
They are implemented using inheritance. Inheritance is explained in detail in polymorphism chapter, for now you just need to know it's a feature which allows to define types as extensions to other types.
std::ios_base
is the base type for all stream types, it holds core stream state (errors, formatting, locale and other stuff).std::basic_ios
extendsstd::ios_base
by adding a stream buffer - a handle to an abstract (hypothetical) I/O device. It's a template to support different character types (CharT
) and their behavior (Traits
).std::basic_istream
andstd::basic_ostream
are primary types for input and output operations, respectively. Out of these types string-streams and file-streams are made.
basic_
is the name prefix C++ standard library uses when a type is a template for generalized implementation. Concrete types are aliases of the basic type with specific template parameters. Examples:
std::string
is an alias ofstd::basic_string<char>
std::wstring
is an alias ofstd::basic_string<wchar_t>
std::ostream
is an alias ofstd::basic_ostream<char>
std::wostream
is an alias ofstd::basic_ostream<wchar_t>
You don't need to understand templates for now. I'm mentioning this because cppreference only documents templates from which these types come from. Don't get surprised when searching for something you land on basic_something with an information what template parameters are - just mentally replace every occurrence of CharT
with the type specified in the alias.
Since wchar_t
has significant use only with Windows-related APIs, pretty much all code you will write and see will use CharT
as char
. Unicode, UTF-8 encoding in particular, uses single byte char
for storing textual data and is by far the most popular text encoding.
Predefined streams
Unix-like operating systems offer 3 predefined streams for every program:
stdin (standard input)
stdout (standard output)
stderr (standard error)
Each program gets them upon startup. By default, they will be connected to the console terminal in which the program is run. Programs which do not have console opened (they usually either have no human interface or only GUI) still have these streams, the data just can not be observed (but could be if they were launched from a terminal or another program opened them through a pipe to collect their output).
C and C++ standard libraries offer global objects which are connected to the operating system's predefined streams:
fd |
stream name |
direction |
C stream object |
C stream type |
C++ stream object |
C++ stream type |
---|---|---|---|---|---|---|
0 |
standard input |
read |
|
|
|
|
1 |
standard output |
write |
|
|
|
|
2 |
standard error |
write |
|
|
|
|
How relevant is this for Windows?
From C and C++ point of view (as a user of the standard library) there is no difference - they just work. Obviously underlying implementation is different - even file descriptors (called file handles on Windows) are designed differently.
Stream redirection
Typical console terminal application combines program's standard output and standard error streams. Still, it doesn't make them the same - their data can be separated using stream redirection.
1 2 3 4 5 6 7 |
#include <iostream> int main() { std::cout << "info message\n"; std::cerr << "error message\n"; } |
Below examples for the Bash shell, though they are compatible with many other shells too:
# redirect stdout to null device $ ./program 1>/dev/null error message # redirect stderr to null device (will display only output) $ ./program 2>/dev/null info message # redirect stderr to a file (output on screen, errors in file) $ ./program 2>errors.txt info message $ cat errors.txt error message # redirect stdin to stdout and then both to a file $ ./program > file.txt 2>&1 $ cat file.txt info message error message
See https://stackoverflow.com/questions/818255/in-the-shell-what-does-21-mean for more examples and explanation.
It's also possible to redirect input:
in such case the program will not wait for keyboard input - std::cin
will eat data from the file and go into failure state upon reaching end of file. This is very useful for testing programs for school/university assignments as it allows to you fully automate program's input.
Buffering
For performance reasons, streams are buffered. The cost of system calls is high (regardless how much data is transferred to/from the system) so streams accumulate data and flush it (pass it to the system) once the buffer is full or a specific thing happened (e.g. text buffer got a newline character).
std::endl
works like 'n' << std::flush
. In majority of situations the flush is redundant and only degrades performance by forcing unnecessary system calls. C++ standard library has a guarantee that standard output is flushed before read operations on standard input. This means you can mix std::cout
with std::cin
without worrying that some data would not be output prior to read operations. For more information, see CppCon 2017: Dietmar Kühl "The End of std::endl" (3min).
Standard error stream is not buffered because errors are generally rare so the buffer would rarely be flushed, delaying output of important information. In the worst case a program could place error information in the buffer, then crash and the error would not be output at all. For this reason error streams output data immediately.
std::clog
and std::wclog
are a buffered standard error stream alternative to std::cerr
and std::wcerr
. As the names suggest, they are intended for logging, which typically are read some time after program execution, thus the lack of need for immediate output allows buffered implementation for increased performance.
Stream limitations
Because streams provide unified way of performing I/O, many device-specific operations are not supported on the stream level of abstraction.
Supported operations:
reading
writing
querying stream state (checking for errors)
Unsupported operations:
size information (different meaning for different things: file - file size, network streams - amount of data transferred)
manipulating device-specific state (e.g. for terminal: moving cursor, clearning output, changing text color, changes in protocol logic for network sockets)
Many C++ beginners ask how to manipulate the terminal (moving cursor, coloring text etc.) but the standard library streams operate on a higher abstraction level and thus do not offer such operations. There are some workarounds (e.g. passing '\r'
to std::cout
will move cursor to the beginning of the line in the terminal, overwriting last line) but they rely on implementation-defined behavior and can easily break (e.g. if executable output is redirected to a file it can become corrupted). So if you really want device-specific behavior you need to use device-specific interface - usually a set of specific functions given by the driver or operating system. This is outside the scope of the C++ standard library and this tutorial.