1,C程序编译步骤
C代码编译成可执行程序经过4步:
1)预处理:宏定义展开、头文件展开、条件编译等,同时将代码中的注释删除,这里并不会检查语法
2)编译:检查语法,将预处理后文件编译生成汇编文件
3)汇编:将汇编文件生成目标文件(二进制文件)
4)链接:C语言写的程序是需要依赖各种库的,所以编译之后还需要把库链接到最终的可执行程序中去
2,gcc编译过程
1) 分步编译
预处理:gcc -E hello.c -o hello.i
编 译: gcc -S hello.i -o hello.s
汇 编:gcc -c hello.s -o hello.o
链 接:gcc hello.o -o hello_elf
选项 含义 -E 只进行预处理 -S(大写) 只进行预处理和编译 -c(小写) 只进行预处理、编译和汇编 -o file 指定生成的输出文件名为 file
文件后缀 含义
.c C 语言文件
.i 预处理后的 C 语言文件
.s 编译后的汇编文件
.o 编译后的目标文件
2) 一步编译
gcc hello.c -o demo(还是经过:预处理、编译、汇编、链接的过程):
3,库文件
3.1 相关文件后缀
.o,是目标文件,相当于windows中的.obj文件
.a为静态库,是好多个.o合在一起,用于静态连接,相当于相当于windows中的 .lib
.so 为共享库,是shared object,用于动态连接的,相当于windows中的.dll
https://www.jianshu.com/p/3fa8e813441a
3.2 GCC链接库文件
1)gcc -l
-l
参数就是用来指定程序要链接的库,-l
参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是 m
,他的库文件名是 libm.so
,很容易看出,把库文件名的头 lib
和尾 .so
去掉就是库名了。好了现在我们知道怎么得到库名,当我们自已要用到一个第三方提供的库名字 libtest.so
,那么我们只要把 libtest.so
拷贝到 /usr/lib
里,编译时加上 -ltest
参数,我们就能用上 libtest.so
库了(当然要用 libtest.so
库里的函数,我们还需要与 libtest.so
配套的头文件)
2)gcc -L
放在 /lib
和 /usr/lib
和 /usr/local/lib
里的库直接用 -l
参数就能链接了,但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用 -l
参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”
,也就是链接程序 ld
在那3个目录里找不到 libxxx.so
。这时另外一个参数 -L
就派上用场了,比如常用的 X11
的库,它在 /usr/X11R6/lib
目录下,我们编译时就要用 -L/usr/X11R6/lib -lX11
参数,-L
参数跟着的是库文件所在的目录名。再比如我们把 libtest.so
放在 /aaa/bbb/ccc
目录下,那链接参数就是 -L/aaa/bbb/ccc -ltest
3)gcc -I
-I
参数是用来指定头文件目录,/usr/include
目录一般是不用指定的,gcc
知道去那里找,但是如果头文件不在 /usr/include
里我们就要用 -I
参数指定了,比如头文件放在 /myinclude
目录里,那编译命令行就要加上 -I/myinclude
参数了,如果不加你会得到一个 "xxxx.h: No such file or directory"
的错误。-I
参数可以用相对路径,比如头文件在当前目录,可以用 -I.
来指定。
一个编译命令
gcc -I/home/zhengshuang/tensorflow_c/include -L/home/zhengshuang/tensorflow_c/lib -ltensorflow test.cpp -o test
https://www.jianshu.com/p/a834e2cd69b3
3,windows下C语言开发环境搭建
https://segmentfault.com/a/1190000014800106
4,C语言中头文件和源文件的关系解析
https://www.cnblogs.com/laojie4321/archive/2012/03/30/2425015.html
5,C语言库函数编写
在windows下C语言的动态库后缀为dll,静态库后缀为lib
https://www.bilibili.com/video/av60294307?from=search&seid=1622683447072032201