I am transitioning a program that uses temporary files from POSIX FILE
to C++ standard library iostreams. What's the correct alternative to mkstemp?
There is no portable C++ way to do it. You need to create a file (which is done automatically when opening a file for writing using an ofstream
) and then remove
it again when you're finished with the file (using the C library function remove). But you can use tmpnam
to generate a name for the file:
#include <fstream>
#include <cstdio>
char filename[L_tmpnam];
std::tmpnam(filename);
std::fstream file(filename);
...
std::remove(filename); //after closing, of course, either by destruction of file or by calling file.close()
There is none. Note that mkstemp
is not part of either C (C99, at least) or C++ standard — it's a POSIX addition. C++ has only tmpfile
and tmpnam
in the C library part.
Boost.IOStreams, however, provides a file_descriptor
device class, which can be used to create a stream operating on what mkstemp
returns.
If I recall correctly, it should look like this:
namespace io = boost::iostreams;
int fd = mkstemp("foo");
if (fd == -1) throw something;
io::file_descriptor device(fd);
io::stream<io::file_descriptor> stream(device);
stream << 42;
If you want a portable C++ solution, you should use unique_path in boost::filesystem :
The unique_path function generates a path name suitable for creating temporary files, including directories. The name is based on a model that uses the percent sign character to specify replacement by a random hexadecimal digit. [Note: The more bits of randomness in the generated path name, the less likelihood of prior existence or being guessed. Each replacement hexadecimal digit in the model adds four bits of randomness. The default model thus provides 64 bits of randomness. This is sufficient for most applications
boost
. It's not built in to my compiler, and I don't want to require people using my source code to download boost. –
Haag unique_path
is as bad as std::tmpnam
due to the race it leads to. –
Mildew tmpnam
has: just tmpnam properly warns you of potential data races while this does not. You need to open(O_EXCL)
, or you need another way to ensure there are no races. –
Sargeant There is no portable C++ way to do it. You need to create a file (which is done automatically when opening a file for writing using an ofstream
) and then remove
it again when you're finished with the file (using the C library function remove). But you can use tmpnam
to generate a name for the file:
#include <fstream>
#include <cstdio>
char filename[L_tmpnam];
std::tmpnam(filename);
std::fstream file(filename);
...
std::remove(filename); //after closing, of course, either by destruction of file or by calling file.close()
© 2022 - 2024 — McMap. All rights reserved.
iostreams
, but there is alwaystmpnam
– Wadaitmpnam
is indeed a good alternative if you can ignore the (small) risk of a file with that name being created in between. – BonSTL
is commonly used to refer to the containers, iterators and algorithms components of the C++ standard library. IOStreams is not really part of the STL, and the STL doesn't need to have an equivalent for every part of the C standard library (which is also available in C++), or for various POSIX functions – Natie