当我要求查看cc的当前版本时,我得到了这个。
$ cc --version
cc (Ubuntu/Linaro 4.7.2-2ubuntu1) 4.7.2
Copyright (C) 2012 Free Software Foundation, Inc.
This is free software; see the source for copying conditions. There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
$
我想知道的是使用c89,c90,c99或c11中的哪一个。
这可以通过输入info gcc
或在线here在gcc手册中深入解释,可用(如果已安装)。 4.7.2手册的相关部分是here。
默认情况下,gcc不符合任何ANSI / ISO C标准。当前的默认值相当于-std=gnu90
,它是具有GNU特定扩展的1989/1990标准。 (不会发布语言标准所要求的某些诊断。)2015年5月22日发布的版本5.1.0将默认值从-std=gnu90
更改为-std=gnu11
,as documented here。
如果您想要标准一致性,可以使用以下任何一种:
-std=c90 -pedantic
-std=c99 -pedantic
-std=c11 -pedantic
-std=c90
也可以拼写-ansi
,-std=c89
或-std=iso9899:1990
。
-std=iso9899:199409
支持C90标准加上1995年的修正案,它增加了一些小功能(所有这些都在C99中)。
-std=c99
也可以拼写-std=c9x
或-std=iso9899:1999
(在标准发布之前使用c9x
这个名字)。 C99的支持是not quite complete,但它很接近。
-std=c11
也可以拼写-std=c0x
或-std=iso9899:2011
(在最终标准出版之前使用c0x
这个名称;错误地认为x
不会超过9)。 C11的支持也不完整;目前的状态是summarized here。
-pedantic
选项使gcc打印违反约束和语法规则所需的诊断信息。在某些情况下,这些诊断仅仅是警告 - 并且没有简单的方法来区分这些警告和语言不需要的其他警告。用-pedantic
替换-pedantic-errors
使gcc将语言违规视为致命错误。
快速的标准历史:
__STDC_VERSION__
和__STDC_LIB_EXT1__
的定义。ANSI没有发布自己的1999或2011标准版本,而是采用ISO标准。
N1256是C99标准的免费提供的草案,其中3个技术勘误表合并。
N1570是C11标准的免费提供的草案。它与已发布的C11标准之间存在一些细微差别,加上一项技术勘误。有关更多详细信息,请参阅my answer到this question。
有用的信息来自info gcc
的gcc6和https://gcc.gnu.org/onlinedocs/gcc-5.4.0/gcc/Standards.html#Standards的gcc5
6.3.1
- 7.3.1
2.1 C Language
==============
The default, if no C language dialect options are given, is
'-std=gnu11'.
2.2 C++ Language
================
The default, if no C++ language dialect options are given, is
'-std=gnu++14'.
5.4.0
2.1 C Language
==============
The default, if no C language dialect options are given, is -std=gnu11
2.2 C++ Language
================
The default, if no C++ language dialect options are given, is -std=gnu++98
对于C,默认模式仍为std = gnu11,但对于C ++,它已从std = gnu ++ 98跳到std = gnu ++ 14
最小的测试程序
如果您想在没有阅读任何手册的情况下凭经验找到它。
C.C
#include <stdio.h>
int main(void) {
#ifdef __STDC_VERSION__
printf("__STDC_VERSION__ = %ld \n", __STDC_VERSION__);
#endif
#ifdef __STRICT_ANSI__
puts("__STRICT_ANSI__");
#endif
return 0;
}
测试:
#!/usr/bin/env bash
for std in c89 c99 c11 c17 gnu89 gnu99 gnu11 gnu17; do
echo $std
gcc -std=$std -o c.out c.c
./c.out
echo
done
echo default
gcc -o c.out c.c
./c.out
结果:
c89
__STRICT_ANSI__
c99
__STDC_VERSION__ = 199901
__STRICT_ANSI__
c11
__STDC_VERSION__ = 201112
__STRICT_ANSI__
c17
__STDC_VERSION__ = 201710
__STRICT_ANSI__
gnu89
gnu99
__STDC_VERSION__ = 199901
gnu11
__STDC_VERSION__ = 201112
gnu17
__STDC_VERSION__ = 201710
default
__STDC_VERSION__ = 201710
结论:默认情况下使用gnu17
:
__STRICT_ANSI__
:为-std=c
定义的GCC扩展,但没有为-std=gnu
定义,请参阅:https://gcc.gnu.org/onlinedocs/cpp/Common-Predefined-Macros.html__STDC_VERSION__
:为每个版本设置的C99 + ANSI C宏。 C89中尚未出现标准尚未定义的情况。C ++
main.cpp中
#include <iostream>
int main(void) {
#ifdef __cplusplus
std::cout << __cplusplus << std::endl;
#endif
#ifdef __STRICT_ANSI__
std::cout << "__STRICT_ANSI__" << std::endl;
#endif
return 0;
}
测试:
#!/usr/bin/env bash
for std in c++98 c++11 c++14 c++17 gnu++98 gnu++11 gnu++14 gnu++17; do
echo $std
g++ -std=$std -o cpp.out cpp.cpp
./cpp.out
echo
done
echo default
g++ -o cpp.out cpp.cpp
./cpp.out
结果:
c++98
199711
__STRICT_ANSI__
c++11
201103
__STRICT_ANSI__
c++14
201402
__STRICT_ANSI__
c++17
201703
__STRICT_ANSI__
gnu++98
199711
gnu++11
201103
gnu++14
201402
gnu++17
201703
default
201402
结论:gnu++14
是默认值:
__cplusplus
:由C ++标准定义的宏,包括C ++ 98及更高版本在Ubuntu 18.10,GCC 8.2.0上测试。 GitHub upstream。
第一行将给出你的GCC版本(4.7.2)
(Ubuntu / Linaro 4.7.2-2ubuntu1)4.7.2
编译代码时,可以通过添加-std=c99
或-std=c99
指定要使用的C / C ++版本...
注意默认情况下使用gnu89
。
需要注意的一点是,gcc的-std =选项不能用于将编译器“沙箱化”为不支持标准C的更高版本的构造。无论是否有-pedantic
都是如此
如果您尝试使用某些C99代码构造进行编译,则不能依赖gcc -std=c89 -pedantic
来提供错误或警告。在某些情况下,它会,但在其他情况下,它不会。例如,它很乐意编译在printf()调用中使用%zu
格式说明符的代码,即使它在C99之前没有添加。
默认gcc命令是ISO C90的GNU方言(包括一些C99功能)。这是C代码的默认值。