编译原理——汇编过程中gcc生成静态库和动态库

系列文章目录

Ubuntu18.04系统的初步使用以及使用gcc和Makefile编译C程序 https://blog.csdn.net/qq_53085291/article/details/126798872



前言

在上一篇文章中,我们已经初步使用了虚拟机,并且已经体会过Ubuntu系统编译程序和Windows系统编译程序的不同。在上一个实验中,我们利用GCC编译器完成了程序编译,但是即使我们已经没有使用IDE那样一键编译程序了,想要更加深入地了解程序编译的过程我们需要GCC背后的故事。


一、目的

我们都知道编译的过程分为三步:预处理、编译和汇编,本文的实验目的就是探究在汇编过程中生成静态库.a和动态库.so的过程。

二、GCC生成静态库和动态库

Hello程序实例

准备过程

1.编写以下代码:
hello.h

#ifndef HELLO_H
#define HELLO_H
void hello(const char *name);
#endif//HELLO_H

hello.c

#include<stdio.h>
void hello(const char *name)
{
	printf("Hello %s\n",name);
}

main.c

#include"hello.h"
int main()
{
	hello("everyone");
	return 0;
}

2.gcc编译得到.o文件

键入以下命令编译源文件:gcc -c hello.c
在这里插入图片描述

静态库操作

我们通常把一些公用函数制作成函数库,供其它程序使用。函数库分为静态库和动态库两种。 静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。动态库在程 序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需 要动态库存在。

1.创建静态库
创建一个名为myhello的静态库文件
创建静态库的工具:ar
静态库文件命名规范:以lib作为前缀,是.a文件

键入命令:ar -crv libmyhello.a hello.o
在这里插入图片描述

2.程序中引用静态库

静态库制作完了,如何使用它内部的函数呢?只需要在使用到这些公用函数的源程序中包 含这些公用函数的原型声明,然后在用 gcc 命令生成目标文件时指明静态库名,gcc 将会从 静态库中将公用函数连接到目标文件中。注意,gcc 会在静态库名前加上前缀 lib,然后追 加扩展名.a 得到的静态库文件名来查找静态库文件。

有以下三种方法在程序中引用静态库

1)键入命令:gcc -o hello main.c -L. -lmyhello

这条命令的语义是从lmyhello静态库中将公用函数hello连接到我们的目标程序main.c文件中。

需要注意的是,自定义的库时,main.c 还可放在-L.和 –lmyhello 之间,但是不能放在它俩之后,否则会提 示 myhello 没定义,但是是系统的库时,如 g++ -o main(-L/usr/lib) -lpthread main.cpp 就不出错。
在这里插入图片描述
2)gcc main.c libmyhello.a -o hello
在这里插入图片描述
3)先生成 main.o
gcc -c main.c
再生成可执行文件:
gcc -o hello main.o libmyhello.a
动态库连接时也可以这样做。
在这里插入图片描述
4)验证静态库的特点
在删掉静态库的情况下,运行可执行文件,发现程序仍旧正常运行,表明静态库跟程序执行没有联系。同时,也表明静态库是在程序编译的时候被连接到代码中的。
在这里插入图片描述

动态库的操作

1.创建动态库
创建动态库的工具:gcc
动态库文件命名规范:以lib作为前缀,是.so文件

gcc -shared -fPIC -o libmyhello.so hello.o
shared:表示指定生成动态链接库,不可省略
-fPIC:表示编译为位置独立的代码,不可省略
命令中的-o一定不能够被省略
在这里插入图片描述

(2). 在程序中执行动态库
gcc -o hello main.c -L. -lmyhello或gcc main.c libmyhello.so -o hello
再运行可执行文件hello,会出现错误
在这里插入图片描述

问题的解决方法:将libmyhello.so复制到目录/usr/lib中。由于运行时,是在/usr/lib中找库文件的。

在这里插入图片描述

3)动态库和静态库的比较

gcc编译得到.o文件 gcc -c hello.c
创建静态库 ar -crv libmyhello.a hello.o
创建动态库 gcc -shared -fPIC -o libmyhello.so hello.o
使用库生成可执行文件 gcc -o hello main.c -L. -lmyhello
执行可执行文件 ./hello

在这里插入图片描述
在执行可执行文件后,我们可以看到当静态库与动态库同时存在的时候,会优先使用动态库。

实例二

代码
sub1.c

float x2x(int a,int b)
{
	float c=0;
	c=a+b;
	return c;
}

sub2.c

float x2y(int a,int b)
{
	float c=0;
	c=a/b;
	return c;
}

sub.h

#ifndef SUB_H
#define SUB_H
float x2x(int a,int b);
float x2y(int a,int b);
#endif

main.c

#include<stdio.h>
#include"sub.h"
void main()
{
	int a,b;
	printf("Please input the value of a:");
	scanf("%d",&a);
	printf("Please input the value of b:");
	scanf("%d",&b);
	printf("a+b=%.2f\n",x2x(a,b));
	printf("a/b=%.2f\n",x2y(a,b));
}

在这里插入图片描述

2.静态库
ar crv libsub.a sub1.o sub2.o
注意:此处用到的文件为.o文件。在这里插入图片描述
3.动态库
gcc -shared -fPIC -o libsub.so sub1.o sub2.o
gcc -o main main.c libsub.so
在这里插入图片描述

实例三

1.准备以下文件

A1.c

#include<stdio.h>
void print1(int arg)
{
	printf("A1 print arg:%d\n",arg);
}

A2.c

#include<stdio.h>
void print2(char *arg)
{
	printf("A2 printf arg:%s\n",arg);
}

A.h

#ifndef A_H
#define A_H
void print1(int);
void print2(char *);
#endif

test.c

#include<stdio.h>
#include"A.h"
int main()
{
	print1(1);
	print2("test");
	return 0;
}

2.静态库
创建静态库
在这里插入图片描述
使用公共函数
在这里插入图片描述
3. 动态库的使用
gcc -shared -fPIC -o libfile.so A1.o A2.o
gcc -o test test.c libfile.so
在这里插入图片描述


总结

通过本次实验我深刻体会到了文件汇编的过程,也逐渐对程序编译过程有了进一步的体会,它并不是像书上总结的短短的预处理、编译、汇编那样一句话概括的过程,而是由多个工具的多种行为协调在一起才得以完成。

参考

https://blog.csdn.net/qq_43279579/article/details/109026927

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值