Linux GCC编译过程及生成静态链接库和动态链接库

一 gcc编译过程 
我们知道gcc是一个强大的编译器,很多Linux下的GNU工具都是用C语言写的,并且用gcc编译的,那么gcc的编译过程是怎样的呢,先来看一个总的流程图,我自己简单画的,凑合着看 
Linux下GCC编译过程及静态链接库和动态链接库 
1首先是源文件经过预加载变成了.i结尾的文件,可以通过-E这个参数来生成这个中间文件,这里主要是把一些include的头文件和一些宏定义,放到源文件中。 
2从预加载的文件经过编译就会变成汇编语言的文件,这一步可以通过-S这个参数来生成这个中间文件 
3从汇编语言的文件通过汇编,就会变成目标代码.o的文件,这一步可以通过-C这个参数来生成这个中间文件 
4最后经过链接,生成最终的可执行文件 
可能这样说比较难懂,我们通过一个例子来说明下: 
我新建了一个hello.c的文件 
Linux下GCC编译过程及静态链接库和动态链接库 
然后我先生成.i结尾的预加载文件 
gcc -E hello.c -o hello.i 
Linux下GCC编译过程及静态链接库和动态链接库 
我们看到这个文件生成了,我们看一下这个文件的内容 
Linux下GCC编译过程及静态链接库和动态链接库 
大家会发现这个文件的内容非常多,但是最后是我们本来的代码,上面的代码都是头文件和一些宏的内容,全加载进来了 
下面我们通过编译生成汇编文件.s 
gcc -S hello.i -o hello.s 
Linux下GCC编译过程及静态链接库和动态链接库 
然后我们看一下这个hello.s 
Linux下GCC编译过程及静态链接库和动态链接库 
看到了我们熟悉的汇编语言 
我们继续,经过汇编器,生成.o的目标文件 
gcc -c hello.s -o hello.o 
Linux下GCC编译过程及静态链接库和动态链接库 
依然来看下内容 
Linux下GCC编译过程及静态链接库和动态链接库 
看到了一堆二进制的感觉 
最后通过链接器,生成可执行文件 
gcc hello.o -o hello 
然后执行 
Linux下GCC编译过程及静态链接库和动态链接库 
当然如果你不想做这么多步骤,直接gcc hello.c -o hello即可完成上面所有的步骤了。

二 静态链接库和动态链接库 
静态链接库就是在程序编译的时候就被加载进来,这样的可执行文件会比较大一些,还不能共享 
动态链接库是在程序执行的时候加载,可共享 
看下面一个例子 
Linux下GCC编译过程及静态链接库和动态链接库 
现在有这么三个文件 
Linux下GCC编译过程及静态链接库和动态链接库 
Linux下GCC编译过程及静态链接库和动态链接库 
Linux下GCC编译过程及静态链接库和动态链接库 
如果直接编译hello.c肯定会报错,因为这里没有main函数,那么我们该怎么处理呢 
1 都编译成.o文件,然后链接,生成可执行文件 
Linux下GCC编译过程及静态链接库和动态链接库

2 利用静态链接库 
在linux下,库文件一般放在/usr/lib和/lib下, 
静态库的名字一般为libxxxx.a,其中xxxx是该lib的名称 
动态库的名字一般为libxxxx.so.major.minor,xxxx是该lib的名称,major是主版本号, minor是副版本号 
静态库的后缀是.a,它的产生分两步 
由源文件编译生成一堆.o,每个.o里都包含这个编译单元的符号表 
ar命令将很多.o转换成.a,成为静态库 
ar rcs libmyhello.a hello.o 
Linux下GCC编译过程及静态链接库和动态链接库 
注意,gcc会在静态库名前加上前缀lib,然后追加扩展名.a得到的静态库文件名来查找静态库文件,因此,我们在写需要连接的库时,只写名字就可以,如libmyhello.a的库,只写:-lmyhello 
Linux下GCC编译过程及静态链接库和动态链接库 
gcc -o hello main.c -static -L. -lmyhello 
-static代表使用静态链接库,-L.代表静态链接库搜索路径 .代表当前路径

3 动态链接库 
gcc -shared -fPIC -c hello.c 
gcc -shared -fPIC -o libmyhello.so hello.o

-share代表是动态链接库 
-fPIC命令行标记告诉GCC产生的代码不要包含对函数和变量具体内存位置的引用,这是因为现在还无法知道使用该消息代码的应用程序会将它连接到哪一段内存地址空间。这样编译出的hello.o可以被用于建立共享链接库 
最后gcc -o hello main.c -L. -lmyhello生成hello可执行文件,注意执行的时候可能会报错,说找不到这个 
libmyhello.so文件,如果放在/lib或者/usr/lib下,那么默认就能找到,如果放在其他目录下,需要编辑/etc/ld.so.conf文件,加入库文件所在目录的路径,然后 
运行ldconfig 目录名字,该命令会重建/etc/ld.so.cache文件即可。

好了,Linux下gcc编译过程,静态链接库和动态链接库就总结到这里,如有问题,欢迎指正,谢谢。

 

转自:https://blog.csdn.net/zjy900507/article/details/88647416


 

在windows下动态链接库是以.dll后缀的文件,二在Linux中,是以.so作后缀的文件。
动态链接库的好处就是节省内存空间。


1、Linux下创建动态链接库
在使用GCC编译程序时,只需加上-shared选项即可,这样生成的执行程序即为动态链接库。
例如有文件:hello.c x.h main.c
编译:gcc hello.c -fPIC -o libhello.so


其中-fPIC选项的作用是:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的,
所以动态载入时是通过代码拷贝的方式来满足不同的调用,而不能达到真正的代码段共享的目的.


将main.c与hello.so动态库
gcc main.c -L. -lhello -o main

一、动态链接库
1.创建hello.so动态库
#include <stdio.h>
void hello(){
    printf("hello world\n");
}
编译:gcc -fPIC -shared hello.c -o libhello.so

2.hello.h头文件
void hello();

3.链接动态库
#include <stdio.h>
#include "hello.h"
 
int main(){
    printf("call hello()");
    hello();
}
编译:gcc main.c -L. -lhello -o main
这里-L的选项是指定编译器在搜索动态库时搜索的路径,告诉编译器hello库的位置。"."意思是当前路径.

3.编译成够后执行./main,会提示:

In function `main':
 
main.c:(.text+0x1d): undefined reference to `hello'
collect2: ld returned 1 exit status
这是因为在链接hello动态库时,编译器没有找到。
解决方法:
sudo cp libhello.so /usr/lib/
这样,再次执行就成功输入:
call hello()

二、静态库

文件有:main.c、hello.c、hello.h
1.编译静态库hello.o: 

gcc hello.c -o hello.o  #这里没有使用-shared

2.把目标文档归档
ar -r libhello.a hello.o  #这里的ar相当于tar的作用,将多个目标打包。
程序ar配合参数-r创建一个新库libhello.a,并将命令行中列出的文件打包入其中。这种方法,如果libhello.a已经存在,将会覆盖现在文件,否则将新创建。

3.链接静态库
gcc main.c -lhello -L. -static -o main
这里的-static选项是告诉编译器,hello是静态库。
或者:
gcc main.c libhello.a -L. -o main
这样就可以不用加-static
4.执行./main

输出:call hello()


三、借助自带的ldd实现程序来分析动态库搜索情况

ldd main

结果:
linux-gate.so.1 =>  (0x00efd000)
libhello.so => /usr/lib/libhello.so (0x00f6b000)
libc.so.6 => /lib/libc.so.6 (0x001a5000)
/lib/ld-linux.so.2 (0x00eb8000)
如果目标程序没有链接动态库,则打印“not a dynamic executable”
————————————————
原文链接:https://blog.csdn.net/a600423444/article/details/7206015

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值