On a cross platform c/c++ project (Win32, Linux, OSX), I need to use the *printf functions to print some variables of type size_t. In some environments size_t's are 8 bytes and on others they are 4. On glibc I have %zd, and on Win32 I can use %Id. Is there an elegant way to handle this?
相关问题
- Sorting 3 numbers without branching [closed]
- Multiple sockets for clients to connect to
- How to compile C++ code in GDB?
- Why does const allow implicit conversion of refere
- thread_local variables initialization
There are really two questions here. The first question is what the correct printf specifier string for the three platforms is. Note that
size_t
is an unsigned type.On Windows, use "
%Iu
".On Linux and OSX, use "
%zu
".The second question is how to support multiple platforms, given that things like format strings might be different on each platform. As other people have pointed out, using
#ifdef
gets ugly quickly.Instead, write a separate makefile or project file for each target platform. Then refer to the specifier by some macro name in your source files, defining the macro appropriately in each makefile. In particular, both GCC and Visual Studio accept a 'D' switch to define macros on the command line.
If your build system is very complicated (multiple build options, generated sources, etc.), maintaining 3 separate makefiles might get unwieldly, and you are going to have to use some kind of advanced build system like CMake or the GNU autotools. But the basic principle is the same-- use the build system to define platform-specific macros instead of putting platform-detection logic in your source files.
The only thing I can think of, is the typical:
and then taking advantage of constant folding:
The
PRIuPTR
macro (from <inttypes.h>) defines a decimal format foruintptr_t
, which should always be large enough that you can cast asize_t
to it without truncating, e.g.Use
boost::format
. It's typesafe, so it'll printsize_t
correctly with%d
, also you don't need to remember to putc_str()
onstd::string
s when using it, and even if you pass a number to%s
or vice versa, it'll work.You just have to find an integer type with the largest storage class, cast the value to it, and then use the appropriate format string for the larger type. Note this solution will work for any type (ptrdiff_t, etc.), not just size_t.
What you want to use is uintmax_t and the format macro PRIuMAX. For Visual C++, you are going to need to download c99-compatible stdint.h and inttypes.h headers, because Microsoft doesn't provide them.
Also see
http://www.embedded.com/columns/technicalinsights/204700432
This article corrects the mistakes in the article quoted by Frederico.
Dan Saks wrote an article in Embedded Systems Design which covered this matter. According to Dan, %zu is the standard way, but few compilers supported this. As an alternative, he recommended using %lu together with an explicit cast of the argument to unsigned long: