gcc是Linux/Unix上非常强大的C语言编译器。并且如今也支持C++的编译。与VC不同,gcc是完全跨平台的。到目前为止很多工具都使用gcc作为其编译环境,比如cocos2d-x ,android的NDK 等等。 学习使用gcc编译器,就是学习它众多的编译参数。就是用连字符- 连接的参数。 |
---|
因为它的参数并不全是单字符的,所以不能使用一个连字符后加多个参数的写法, 要每个参数前面都要加一个连字符
gcc | g++ | vc | |
---|---|---|---|
main | √ | x | x |
void main | x | x | √ |
int main | √ | √ | √ |
直接在gcc后面加上要编译的C语言源文件
gcc hello.c
这种默认的情况下会生成一个名为a.out
的可执行文件。
注意,在Linux系统下,后缀名真的不是很重要的事,甚至没有后缀名都可以。这些都没有关系,比如你写一个shell的脚本,其实加不加后缀都是可以运行的。此时后缀的目的是为了便于用户管理和区分文件而已。By the way,如果你执行
./hello.c
,那么系统会把它也当做shell脚本来运行,根本不管什么后缀。
最常用的编译选项,用于指定要生成的可执行文件的名称。
gcc -o hello hello.c
需要体会的一点是:gcc对于参数的位置无要求!比如也可以写作
gcc hello.c -o hello
惟一要注意的就是-o
后面一定要紧跟要生成的可执行文件名。
编译出的可执行文件,在Windows下就是exe(executive)。但Linux下,后缀名无限制。一般不指定后缀就可以了。
用于给生成的可执行文件加上调试信息,只有这样才可使用gdb调试。
gcc -g -o hello hello.c
同样参数-g
的也可写在hello.c后面。
终止链接器的运行,输出文件为汇编后的目标文件*.o
gcc -c hello.c
生成文件为hello.o
对源码进行优化,使编译出的程序,运行效率更高。注意是大写的英语字母O
,不是阿拉伯数字0
gcc -O2 hello.c -o hello
O
是Optimize之意。同样还有O1
,但是优化效果不如O2
,缺省是O0
给编译的源文件传递一个宏。
gcc a.c -DHELLO -DWORLD=10
相当于:
//在a.c中定义了
# define HELLO
# define WORLD 10
gcc -E 指示gcc对源代码进行预处理,结果直接输出到终端。
实际上和命令
cpp 源文件
相同。cpp是c预处理器的意思,而非c++的意思。
包含自定义头文件的路径
生成汇编代码以.s为后缀。
默认是AT&T汇编语法,加选项masm=intel可生成Intel语法的汇编。
gcc -S -masm=intel test.c
静态链接库是后缀名为.a的文件。它有多个后缀为你.o的目标文件组成。
使用-c
参数可以使编译在链接前终止,所以生成的是源文件对应的目标文件。
gcc -c addvec.c multvec.c
ar rcs libvector.a addvec.o multvec.o
ar
是archive档案的缩写。上面命令生成了.a的静态库文件,在链接时,要如下:
gcc -c main2.c
gcc -static -o p2 main2.o ./libvector.a
后缀名为.so的是动态共享链接库文件,其中的s就是shared共享的意思,如下命令:
gcc -shared -fPIC -o libvector.so addvec.c multvec.c
gcc -o p2 main2.c ./libvector.so
完成了生成.so以及链接.so的操作。-fPIC指示生成与位置无关的代码。
gcc -o temp temp.c -lm
-lm选项,在编译时会进入系统库路径搜索,链接“数学库”。常用的库会自动链接,无需指定。
系统缺省的库路径为:/lib、/usr/lib、/usr/local/lib、/usr/lib64。
数学库的文件可能为libm-2.1.2.so去掉lib和后面的版本号就只剩下m了。
如果该库不在系统缺省路径下(比如第三方库,自定义的库),还要使用**-L**选项指定路径。
-L/home/jelly/mylib
-L选项指定的是在编译期间库的搜索路径,然而如果是动态库的话,在运行时加载库,此时只会搜索默认库路径。
此时需在编译时加上**-Wl,-rpath=**后面指定路径。注意这不是两个选项。
g++ -I ../include -Wl,-rpath=../util/ -lwang -L../util/ client.c -o client
另外一种方案是修改配置文件:/etc/ld.so.conf
。然后需要ldconfig命令(root)更新。如果无root权限,请使用上一种方案。
有时候连接通过,但运行时出错:未定义符号XXX。请查看一下是否正确保护库的路径,如果确认无误,仍有错。
可以使用**ldd**命令检查链接的库的绝对路径。
有时候自己写的库和系统的库重名就会出现该错误。
默认的情况下,编译器是以c89的标准编译的。使用c99则:
gcc -std=c99 hello.c
-I
选项指定头文件的位置。一般和L选项联用。比如:
gcc mysql_test.c -I /usr/local/include/mysql -L/usr/local/lib -lmysqlclient -o test
gcc -M main.c
以makefile风格显示源文件的依赖关系。会列出所有包含的所有头文件。会列出标准库的头文件。 如果不想显示标准库的头文件,请使用**-MM选项代替-M**。
适用于g++。C++语言因为各种临时对象的问题,所以编译器通常会自行进行优化,比如NRV优化(O0已存在该优化),会减少几次拷贝构造函数的调用过程。如果你想关闭这个优化:
g++ -fno-elide-constructors hello.cpp