gcc编译程序经历的步骤
gcc编译器从拿到一个c源文件到生成一个可执行程序,中间一共经历了四个步骤:
gcc工作的流程:
预处理
生成汇编文件
生成目标代码
生成可以执行文件
gcc常用选项:
-o file 指定生成的输出文件名为file
-g 包含调试信息
-On n=0~3 编译优化,n越大优化得越多
-Wall 提示更多警告信息
静态连接和动态连接
链接分为两种:静态链接、动态链接。
1)静态链接
静态链接:由链接器在链接时将库的内容加入到可执行程序中。
优点:对运行环境的依赖性较小,具有较好的兼容性
缺点:生成的程序比较大,需要更多的系统资源,在装入内存时会消耗更多的时间。库函数有了更新,必须重新编译应用程序
2)动态链接
动态链接:连接器在链接时仅仅建立与所需库函数的之间的链接关系,在程序运行时才将所需资源调入可执行程序。
优点:
- 在需要的时候才会调入对应的资源函数
- 简化程序的升级;有着较小的程序体积
- 实现进程之间的资源共享(避免重复拷贝)
缺点:
- 依赖动态库,不能独立运行
- 动态库依赖版本问题严重
静态、动态编译对比
系统默认采用动态链接的方式进行编译程序,若想采用静态编译,加入-static参数。
静态库和动态库简介
“程序库”:包含了数据和执行代码的文件。其不能单独执行,可以作为其它执行程序的一部分来完成某些功能。
库的存在可以使得程序模块化,可以加快程序的再编译,可以实现代码重用,可以使得程序便于升级。
程序库可分静态库(static library)和共享库(shared library)。
静态库制作
静态库可以认为是一些目标代码的集合,是在可执行程序运行前就已经加入到执行码中,成为执行程序的一部分
前缀:lib
库名称:自己定义即可
后缀:.a
libxxx.a
制作方式:
- 将c源文件生成对应的.o文件
- 使用打包工具ar将准备好的.o文件打包为.a文件 libtest.a
在使用ar工具是时候需要添加参数:rcs
r更新
c创建
s建立索引
静态库使用
静态库制作完成之后,需要将**.a文件和头文件**一起发布给用户。
gcc test.c -L./ -I./ -ltest -o test
-L:表示**要连接的库所在目录**
-I./: I(大写i) 表示指定头文件的目录为当前目录
-l(小写L):**指定链接时需要的库,去掉前缀和后缀**
动态库制作
共享库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例,规避了空间浪费问题。
动态库在程序运行是才被载入,也解决了静态库对程序的更新、部署和发布页会带来麻烦。用户只需要更新动态库即可,增量更新。
前缀:lib
库名称:自己定义即可
后缀:.so
libxxx.so
制作过程:
- 生成目标文件,此时要加编译选项:-fPIC(fpic)。-fPIC 创建与地址无关的编译程序(pic,position independent code),是为了能够在多个应用程序间共享。
- 生成共享库,此时要加链接器选项: -shared(指定生成动态链接库)
- 通过nm命令查看对应的函数
- ldd查看可执行文件的依赖的动态库
动态库测试
gcc test.c -L. -I. -ltest (-I. 大写i -ltest 小写L)
当系统加载可执行代码时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径。此时就需要系统动态载入器(dynamic linker/loader)。
如何让系统找到动态库:
1.拷贝自己制作的共享库到**/lib或者/usr/lib**(不能是/lib64目录)
2.临时设置LD_LIBRARY_PATH:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
2.永久设置,把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径,设置到**~/.bashrc**或者 /etc/profile文件中。
source ~/.bashrc:使环境变量生效
3.将其添加到 /etc/ld.so.conf文件中
编辑/etc/ld.so.conf文件,加入库文件所在目录的路径
运行sudo ldconfig -v,该命令会重建/etc/ld.so.cache文件