如果有一些跨平台的C/ c++代码应该在Mac OS X, iOS, Linux, Windows上编译,我如何在预处理器过程中可靠地检测它们?
当前回答
2021年1月5日:由于@Sadap的评论,链接更新。
这是一个必然的答案:这个网站上的人已经花时间为每个OS/编译器对定义了宏表。
例如,你可以看到_WIN32在Windows Cygwin (POSIX)上没有定义,而在Windows Cygwin(非POSIX)上定义了编译,在MinGW上定义了所有可用的编译器(Clang, GNU, Intel等)。
不管怎样,我发现这些表格非常有用,我想在这里分享。
其他回答
大多数编译器都使用预定义的宏,你可以在这里找到列表。GCC编译器预定义宏可以在这里找到。 下面是gcc的一个例子:
#if defined(WIN32) || defined(_WIN32) || defined(__WIN32__) || defined(__NT__)
//define something for Windows (32-bit and 64-bit, this part is common)
#ifdef _WIN64
//define something for Windows (64-bit only)
#else
//define something for Windows (32-bit only)
#endif
#elif __APPLE__
#include <TargetConditionals.h>
#if TARGET_IPHONE_SIMULATOR
// iOS, tvOS, or watchOS Simulator
#elif TARGET_OS_MACCATALYST
// Mac's Catalyst (ports iOS API into Mac, like UIKit).
#elif TARGET_OS_IPHONE
// iOS, tvOS, or watchOS device
#elif TARGET_OS_MAC
// Other kinds of Apple platforms
#else
# error "Unknown Apple platform"
#endif
#elif __ANDROID__
// Below __linux__ check should be enough to handle Android,
// but something may be unique to Android.
#elif __linux__
// linux
#elif __unix__ // all unices not caught above
// Unix
#elif defined(_POSIX_VERSION)
// POSIX
#else
# error "Unknown compiler"
#endif
所定义的宏取决于将要使用的编译器。
_WIN64 #ifdef可以嵌套到_WIN32 #ifdef中,因为_WIN32甚至是在针对Windows x64版本时定义的。如果某些头包含对两者都是公共的,则可以防止代码重复 (没有下划线的WIN32允许IDE突出显示正确的代码分区)。
正如Jake指出的,TARGET_IPHONE_SIMULATOR是TARGET_OS_IPHONE的子集。
另外,TARGET_OS_IPHONE是TARGET_OS_MAC的子集。
所以一个更好的方法可能是:
#ifdef _WIN64
//define something for Windows (64-bit)
#elif _WIN32
//define something for Windows (32-bit)
#elif __APPLE__
#include "TargetConditionals.h"
#if TARGET_OS_IPHONE && TARGET_OS_SIMULATOR
// define something for simulator
// (although, checking for TARGET_OS_IPHONE should not be required).
#elif TARGET_OS_IPHONE && TARGET_OS_MACCATALYST
// define something for Mac's Catalyst
#elif TARGET_OS_IPHONE
// define something for iphone
#else
#define TARGET_OS_OSX 1
// define something for OSX
#endif
#elif __linux
// linux
#elif __unix // all unices not caught above
// Unix
#elif __posix
// POSIX
#endif
注意上面检查了TARGET_OS_SIMULATOR宏,因为TARGET_IPHONE_SIMULATOR宏从iOS 14开始就被弃用了。
2021年1月5日:由于@Sadap的评论,链接更新。
这是一个必然的答案:这个网站上的人已经花时间为每个OS/编译器对定义了宏表。
例如,你可以看到_WIN32在Windows Cygwin (POSIX)上没有定义,而在Windows Cygwin(非POSIX)上定义了编译,在MinGW上定义了所有可用的编译器(Clang, GNU, Intel等)。
不管怎样,我发现这些表格非常有用,我想在这里分享。
推荐文章
- 为什么这个结合赋值和相等检查的if语句返回true?
- cplusplus.com给出的错误、误解或坏建议是什么?
- C多行宏:do/while(0) vs作用域块
- 找出质数最快的算法是什么?
- time_t最终的类型定义是什么?
- 我需要显式处理负数或零时,总和平方数字?
- c++枚举类可以有方法吗?
- 函数名周围的括号是什么意思?
- 用C语言创建自己的头文件
- 格式化IO函数(*printf / *scanf)中的转换说明符%i和%d之间的区别是什么?
- 将析构函数设为私有有什么用?
- main()中的Return语句vs exit()
- 为什么c#不提供c++风格的'friend'关键字?
- 如果不是内存地址,C指针到底是什么?
- 在函数的签名中添加关键字