如何确定操作系统是否为C中的POSIX?
相关问题如何确定操作系统是否为C中的POSIX?,c,gcc,posix,C,Gcc,Posix,相关问题 我怎样才能知道我在GCC或ANSI C中运行的是什么操作系统 如果我能知道我是否在POSIX上运行,我会很好 更新: 无论它是在编译时还是在运行时,对我来说都没有区别。我在调试例程中使用它,所以性能并不是那么重要 我在找路径分隔符。Windows&Unix/Linux/BSD就可以了 我正试图在路径上找到basename。我找到了一些解决方案,但这些解决方案包含了很多我不想要的内容。我要修改一下 我在Mac OS X 10.4.11上,执行了以下命令并获得以下输出: mac $
我怎样才能知道我在GCC或ANSI C中运行的是什么操作系统 如果我能知道我是否在POSIX上运行,我会很好
更新: 无论它是在编译时还是在运行时,对我来说都没有区别。我在调试例程中使用它,所以性能并不是那么重要 我在找路径分隔符。Windows&Unix/Linux/BSD就可以了 我正试图在路径上找到basename。我找到了一些解决方案,但这些解决方案包含了很多我不想要的内容。我要修改一下
我在Mac OS X 10.4.11上,执行了以下命令并获得以下输出:
mac $ touch myfile.c
mac $ gcc -std=c99 -E -dM myfile.c
#define __DBL_MIN_EXP__ (-1021)
#define __FLT_MIN__ 1.17549435e-38F
#define __CHAR_BIT__ 8
#define __WCHAR_MAX__ 2147483647
#define __DBL_DENORM_MIN__ 4.9406564584124654e-324
#define __FLT_EVAL_METHOD__ 0
#define __DBL_MIN_10_EXP__ (-307)
#define __FINITE_MATH_ONLY__ 0
#define __GNUC_PATCHLEVEL__ 1
#define __SHRT_MAX__ 32767
#define __LDBL_MAX__ 1.79769313486231580793728971405301e+308L
#define __APPLE_CC__ 5367
#define __UINTMAX_TYPE__ long long unsigned int
#define __LDBL_MAX_EXP__ 1024
#define __SCHAR_MAX__ 127
#define __USER_LABEL_PREFIX__ _
#define __STDC_HOSTED__ 1
#define __LDBL_HAS_INFINITY__ 1
#define __DBL_DIG__ 15
#define __FLT_EPSILON__ 1.19209290e-7F
#define __LDBL_MIN__ 2.00416836000897277799610805135016e-292L
#define __ppc__ 1
#define __strong
#define __APPLE__ 1
#define __DECIMAL_DIG__ 33
#define __LDBL_HAS_QUIET_NAN__ 1
#define __DYNAMIC__ 1
#define __GNUC__ 4
#define __DBL_MAX__ 1.7976931348623157e+308
#define __DBL_HAS_INFINITY__ 1
#define __STRICT_ANSI__ 1
#define __weak
#define __DBL_MAX_EXP__ 1024
#define __LONG_LONG_MAX__ 9223372036854775807LL
#define __GXX_ABI_VERSION 1002
#define __FLT_MIN_EXP__ (-125)
#define __DBL_MIN__ 2.2250738585072014e-308
#define __DBL_HAS_QUIET_NAN__ 1
#define __REGISTER_PREFIX__
#define __NO_INLINE__ 1
#define _ARCH_PPC 1
#define __FLT_MANT_DIG__ 24
#define __VERSION__ "4.0.1 (Apple Computer, Inc. build 5367)"
#define __BIG_ENDIAN__ 1
#define __SIZE_TYPE__ long unsigned int
#define __FLT_RADIX__ 2
#define __LDBL_EPSILON__ 4.94065645841246544176568792868221e-324L
#define __NATURAL_ALIGNMENT__ 1
#define __FLT_HAS_QUIET_NAN__ 1
#define __FLT_MAX_10_EXP__ 38
#define __LONG_MAX__ 2147483647L
#define __FLT_HAS_INFINITY__ 1
#define __STDC_VERSION__ 199901L
#define _BIG_ENDIAN 1
#define __LDBL_MANT_DIG__ 106
#define __WCHAR_TYPE__ int
#define __FLT_DIG__ 6
#define __INT_MAX__ 2147483647
#define __LONG_DOUBLE_128__ 1
#define __FLT_MAX_EXP__ 128
#define __DBL_MANT_DIG__ 53
#define __WINT_TYPE__ int
#define __LDBL_MIN_EXP__ (-968)
#define __MACH__ 1
#define __LDBL_MAX_10_EXP__ 308
#define __DBL_EPSILON__ 2.2204460492503131e-16
#define __INTMAX_MAX__ 9223372036854775807LL
#define __FLT_DENORM_MIN__ 1.40129846e-45F
#define __PIC__ 1
#define __FLT_MAX__ 3.40282347e+38F
#define __FLT_MIN_10_EXP__ (-37)
#define __INTMAX_TYPE__ long long int
#define __GNUC_MINOR__ 0
#define __DBL_MAX_10_EXP__ 308
#define __LDBL_DENORM_MIN__ 4.94065645841246544176568792868221e-324L
#define __PTRDIFF_TYPE__ int
#define __LDBL_MIN_10_EXP__ (-291)
#define __LDBL_DIG__ 31
#define __POWERPC__ 1
mac $
单个UNIX规范需要存在,它可以告诉您POSIX版本(通过
\u POSIX\u version
宏)
但是,如果您还不知道自己实际上是在UNIX上编译,那么如何包含unistd.h
这就是方便的地方。根据它的说法,测试\uuuuuunix\uuuuuu
的存在性或评估为true应该告诉您该系统是一个unix系统。因此:
#ifdef __unix__
/* Yes it is a UNIX because __unix__ is defined. */
#include <unistd.h>
/* You can find out the version with _POSIX_VERSION.
..
.. */
#endif
要获取系统上特定于系统的预定义宏的列表,可以执行:
cpp -dM /dev/null
例如,我的GNU/Linux系统还额外定义了\uuuu Linux\uuuu
和\uuuu GNU\uuu Linux\uuuu
,除了\uuuu unix\uuu
和其他一些东西
您必须查看的另一个有用文档是 它在某种程度上与我上面描述的类似
编辑:既然您确实想做这一切,因为您想决定使用哪个目录分隔符,请查看。它说: 注意Windows API中的文件I/O函数将“/”转换为“\”作为一部分 将名称转换为NT样式名称的步骤,除非使用 “\?\”前缀,详见以下章节
我不在Windows上编程,也不太了解它,所以我不能说我指望它。以下是我的结论:
#include <stdio.h>
/*
NAME
basename -- return pointer to last component of a pathname
Distribution: This function is in the public domain.
Origin of function:
http://www.koders.com/c/fidEB79B7607A210C3BB7B813E793465F9D469AE912.aspx
SYNOPSIS
char *basename (const char *name)
DESCRIPTION
Given a pointer to a string containing a typical pathname
(/usr/src/cmd/ls/ls.c for example), returns a pointer to the
last component of the pathname ("ls.c" in this case).
Restrictions:
Presumes a UNIX or DOS/Windows style path with UNIX or DOS/Windows
style separators.
*/
/*
NAME:
basename:
Function:
return pointer to last component of a pathname
Distribution:
This function is in the public domain.
Origin of function:
http://www.koders.com/c/fidEB79B7607A210C3BB7B813E793465F9D469AE912.aspx
SYNOPSIS:
char *basename (const char *name)
DESCRIPTION:
Given a pointer to a string containing a typical pathname
(/usr/src/cmd/ls/ls.c for example), returns a pointer to the
last component of the pathname ("ls.c" in this case).
Restrictions:
Presumes a UNIX or DOS/Windows style path with UNIX or
DOS/Windows style separators.
Windows volumes are only a-zA-Z.
The original code suggests ISALPHA.
*/
char * basename (const char *name)
{
const char *base;
// predefined OS symbols
// http://sourceforge.net/apps/mediawiki/predef/index.php?title=Operating_Systems#UNIX_Environment
#ifndef DIR_SEPARATOR
# define DIR_SEPARATOR '/'
#endif
#ifndef DIR_SEPARATOR_2
# define DIR_SEPARATOR_2 '\\'
#endif
// Check if we are running Unix like os
// else assume Windows. Note if we guess wrong, it's not
// so bad because Windows includes the Unix separator.
#if defined (__unix__) || (defined (__APPLE__) && defined (__MACH__))
# define IS_DIR_SEPARATOR(ch) ((ch) == DIR_SEPARATOR)
#else
# define IS_DIR_SEPARATOR(ch) \
(((ch) == DIR_SEPARATOR) || ((ch) == DIR_SEPARATOR_2))
/* Skip over the disk name in MSDOS pathnames. */
if (isalpha(name[0]) && name[1] == ':')
name += 2;
#endif
for (base = name; *name; name++)
{
if (IS_DIR_SEPARATOR (*name))
{
base = name + 1;
}
}
return (char *) base;
}
int main (int argc, const char * argv[]) {
/* Return the basename of a pathname. */
#define S1 "/usr/src/cmd/ls/ls.c"
#define S2 "/usr/src/cmd/ls/abc"
#define S3 "a:/usr/src/cmd/ls/def"
#define S4 "ghi"
#define S5 "jkl.txt"
#define S6 "d:\\usr\\src\\cmd\\mno.txt"
#define S7 "d:pqm.txt"
printf(S1 " \t is %s\n",basename(S1));
printf(S2 " \t is %s\n",basename(S2));
printf(S3 " \t is %s\n",basename(S3));
printf(S4 " \t is %s\n",basename(S4));
printf(S5 " \t is %s\n",basename(S5));
printf(S6 " \t is %s\n",basename(S6));
printf(S7 " \t is %s\n",basename(S7));
return 0;
#包括
/*
名称
basename--返回指向路径名最后一个组件的指针
分发:此功能位于公共域中。
功能来源:
http://www.koders.com/c/fidEB79B7607A210C3BB7B813E793465F9D469AE912.aspx
提要
字符*基本名称(常量字符*名称)
描述
给定一个指向包含典型路径名的字符串的指针
(/usr/src/cmd/ls/ls.c)返回指向
路径名的最后一个组件(在本例中为“ls.c”)。
限制:
假定UNIX或DOS/Windows具有UNIX或DOS/Windows样式的路径
样式分隔符。
*/
/*
姓名:
基本名称:
功能:
返回指向路径名最后一个组件的指针
分布:
此功能属于公共领域。
功能来源:
http://www.koders.com/c/fidEB79B7607A210C3BB7B813E793465F9D469AE912.aspx
简介:
字符*基本名称(常量字符*名称)
说明:
给定一个指向包含典型路径名的字符串的指针
(/usr/src/cmd/ls/ls.c)返回指向
路径名的最后一个组件(在本例中为“ls.c”)。
限制:
假定UNIX或DOS/Windows样式的路径与UNIX或
DOS/Windows样式分隔符。
Windows卷仅为a-zA-Z。
原始代码建议使用ISALPHA。
*/
字符*基本名称(常量字符*名称)
{
常量字符*基;
//预定义的操作系统符号
// http://sourceforge.net/apps/mediawiki/predef/index.php?title=Operating_Systems#UNIX_Environment
#ifndef定向分离器
#定义目录分隔符“/”
#恩迪夫
#ifndef方向分离器2
#定义目录分隔符2'\\'
#恩迪夫
//检查我们是否运行类似Unix的操作系统
//如果我们猜错了,那就错了
//这太糟糕了,因为Windows包含Unix分隔符。
#如果定义(uuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuu
#定义IS_DIR_分隔符(ch)((ch)=DIR_分隔符)
#否则
#定义IS_DIR_分隔符(ch)\
(((ch)=方向分隔符)| |((ch)=方向分隔符|2))
/*跳过MSDOS路径名中的磁盘名*/
if(isalpha(名称[0])&&name[1]==':')
名称+=2;
#恩迪夫
for(base=name;*name;name++)
{
if(IS_DIR_分隔符(*名称))
{
基数=名称+1;
}
}
返回(字符*)基;
}
int main(int argc,const char*argv[]{
/*返回路径名的basename*/
#定义S1“/usr/src/cmd/ls/ls.c”
#定义S2“/usr/src/cmd/ls/abc”
#定义S3“a:/usr/src/cmd/ls/def”
#定义S4“ghi”
#定义S5“jkl.txt”
#定义S6“d:\\usr\\src\\cmd\\mno.txt”
#定义S7“d:pqm.txt”
printf(S1“\t是%s\n”,basename(S1));
printf(S2“\t是%s\n”,basename(S2));
printf(S3“\t是%s\n”,basename(S3));
printf(S4“\t是%s\n”,basename(S4));
printf(S5“\t是%s\n”,basename(S5));
printf(S6“\t是%s\n”,basename(S6));
printf(S7“\t是%s\n”,basename(S7));
返回0;
}如果找到了
unistd.h
,则可以使用来发现unistd.h
的存在,然后在生成的config.h
中添加一行\define HAVE\u unistd\u h1
,但我发现autoconf有点难以使用,而且非常过时
如果您使用的是cmake,您可以用同样的方法解决它。您可以在中创建一个config.h.in,其中包含以下内容:
#ifndef CONFIG_H
#define CONFIG_H
#cmakedefine HAVE_UNISTD_H 1
#endif
project(MyApp)
include(CheckIncludeFiles)
check_include_file("unistd.h" HAVE_UNISTD_H)
configure_file(config.h.in config.h @ONLY)
add_executable(${PROJECT_NAME} main.c)
#ifndef CONFIG_H
#define CONFIG_H
#define HAVE_UNISTD_H 1
#endif
您的项目的CMakeLists.txt
如下所示:
#ifndef CONFIG_H
#define CONFIG_H
#cmakedefine HAVE_UNISTD_H 1
#endif
project(MyApp)
include(CheckIncludeFiles)
check_include_file("unistd.h" HAVE_UNISTD_H)
configure_file(config.h.in config.h @ONLY)
add_executable(${PROJECT_NAME} main.c)
#ifndef CONFIG_H
#define CONFIG_H
#define HAVE_UNISTD_H 1
#endif
然后从命令行生成:
cmake . -G"Unix Makefiles"
或生成Xcode项目(仅限OSX):
或生成visual studio 2013解决方案项目(仅限Windows):
cmake生成器=史诗般的胜利
如果您的操作系统是POSIX,那么生成的config.h
应该如下所示:
#ifndef CONFIG_H
#define CONFIG_H
#cmakedefine HAVE_UNISTD_H 1
#endif
project(MyApp)
include(CheckIncludeFiles)
check_include_file("unistd.h" HAVE_UNISTD_H)
configure_file(config.h.in config.h @ONLY)
add_executable(${PROJECT_NAME} main.c)
#ifndef CONFIG_H
#define CONFIG_H
#define HAVE_UNISTD_H 1
#endif
否则,它将如下所示:
#ifndef CONFIG_H
#define CONFIG_H
/* #undef HAVE_UNISTD_H */
#endif
然后,您就可以自由访问您的可信生成的config.h
:
#include "config.h"
#if HAVE_UNISTD_H
# include <unistd.h>
#endif
int main (int argc, const char * argv[])
{
#if defined(_POSIX_VERSION)
/* POSIX code here */
#else
/* non-POSIX code here */
#endif
return 0;
}
#包括“config.h”
#如果你有
#包括
#恩迪夫
int m