简体   繁体   中英

Read binary data from std::cin

将二进制(非格式化)数据从std::cin读取到stringstringstream的最简单方法是什么?

std::cin is not opened with ios_binary. If you must use cin, then you need to reopen it, which isn't part of the standard.

Some ideas here: https://comp.unix.programmer.narkive.com/jeVj1j3I/how-can-i-reopen-std-cin-and-std-cout-in-binary-mode

Once it's binary, you can use cin.read() to read bytes. If you know that in your system, there is no difference between text and binary (and you don't need to be portable), then you can just use read without worrying.

For windows, you can use the _setmode function in conjunction with cin.read() , as already mentioned.

_setmode(_fileno(stdin), _O_BINARY);
cin.read(...);

See solution source here: http://talmai-oliveira.blogspot.com/2011/06/reading-binary-files-from-cin.html

On a Unix/POSIX system, you can use the cin.get() method to read byte-by-byte and save the data into a container like a std::vector<unsigned int> , or you can use cin.read() in order to read a fixed amount of bytes into a buffer. You could also use cin.peek() to check for any end-of-data-stream indicators.

Keep in mind to avoid using the operator>> overload for this type of operation ... using operator>> will cause breaks to occur whenever a delimiter character is observed, and it will also remove the delimiting character from the stream itself. This would include any binary values that are equivalent to a space, tab, etc. Thus the binary data your end up storing from std::cin using that method will not match the input binary stream byte-for-byte.

cin.read would store a fixed number of bytes, without any logic searching for delimiters of the type that @Jason mentioned.

However, there may still be translations active on the stream, such as CRLF -> NL, so it still isn't ideal for binary data.

All predefined iostream objects are obligated to be bound to corresponding C streams:

The object cin controls input from a stream buffer associated with the object stdin , declared in <cstdio> .

http://eel.is/c++draft/narrow.stream.objects

and thus the method of obtaining binary data is same as for C:

Basically, the best you can really do is this:

freopen(NULL, "rb", stdin);

This will reopen stdin to be the same input stream, but in binary mode. In the normal mode, reading from stdin on Windows will convert \r\n (Windows newline) to the single character ASCII 10. Using the "rb" mode disables this conversion so that you can properly read in binary data.

https://stackoverflow.com/a/1599093/6049796

With windows/mingw/msys/bash, if you need to pipe different commands with binary streams in between, you need to manipulate std::cin and std::cout as binary streams.

The _setmode solution from Mikhail works perfectly.

Using MinGW, the neaded headers are the following:

#include <io.h>
#include <fcntl.h> 

cplusplus.com :

  • Unformatted input

    Most of the other member functions of the istream class are used to perform unformatted input, ie no interpretation is made on the characters got form the input. These member functions can get a determined number of characters from the input character sequence ( get , getline , peek , read , readsome )...

As Lou Franco pointed out, std::cin isn't opened with std::ios_base::binary, but one of those functions might get you close to the behavior you're looking for.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM