C语言编译过程及相关文件

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,库文件

  动态链接,在可执行文件装载时或运行时,由操作系统的装载程序加载库。大多数操作系统将解析外部引用(比如库)作为加载过程的一部分。在这些系统上,可执行文件包含一个叫做import directory的表,该表的每一项包含一个库的名字。根据表中记录的名字,装载程序在硬盘上搜索需要的库,然后将其加载到内存中预先不确定的位置,之后根据加载库后确定的库的地址更新可执行程序。可执行程序根据更新后的库信息调用库中的函数或引用库中的数据。这种类型的动态加载成为装载时加载 ,被包括Windows和Linux的大多数系统采用。装载程序在加载应用软件时要完成的最复杂的工作之一就是加载时链接。(还有一种动态加载方式叫运行时链接,但该方式要慢得多,对应用的性能有负面影响。现代操作系统已经很少使用运行时链接。)
  可以动态链接的库,在Windows上是dynamic link library (DLL),在UNIX或Linux上是Shared Library。库文件是预先编译链接好的可执行文件,存储在计算机的硬盘上。大多数情况下,同一时间多个应用可以使用一个库的同一份拷贝,操作系统不需要加载这个库的多个实例。

3.1 相关文件后缀

Windows 和 Linux 的加载时链接是由操作系统来完成的,格式在不同的系统下有不同的区别,但是原理还是一样的。
 
linux下文件的类型是不依赖于其后缀名的,但一般来讲:

.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

6,VS下创建自己的的静态库并使用

7,QT下创建自己的静态库并使用

https://blog.csdn.net/ALONE_WORK/article/details/78817547

posted @ 2019-11-25 17:01  xdyixia  阅读(1761)  评论(0编辑  收藏  举报