一、学习理解用gcc生成静态库和动态库和静态库.a与.so库文件的生成与使用
1 文件准备
创建文件夹test1,进入并写入以下几个程序
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 <stdlib.h>
#include "A.h"
int main()
{
print1(1);
print2("test");
exit(0);
}
2、静态库的生成和编译
生成静态库文件 ar crv libafile.a A1.o A2.o
使用静态库文件libafile.a 编译生成程序gcc -o test test.c libafile.a
3、动态库的生成和编译
生成对象文件(添加命令"-fpic",否则在生成.so文件时会出错)gcc -c -fpic A1.c A2.c
生成动态库.so文件 gcc -shared *.o -o libsofile.so
使用.so文件,创建可执行程序gcc -o test test.c libsofile.so,执行./test发现报错
使用ldd命令查看,发现找不到对应的libsofile.so文件,报错信息如下。百度可查得这是由于linux 自身系统设定的相应的设置的原因,即其只在/lib and /usr/lib 下搜索对应的.so 文件,故需将对应so 文件拷贝到对应路径。
$ ldd test
linux-vdso.so.1 (0x00007ffec186c000)
libsofile.so => not found
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007efe2dbac000)
/lib64/ld-linux-x86-64.so.2 (0x00007efe2dde3000)
$
执行拷贝命令sudo cp libsofile.so /usr/lib,再次执行目标文件可以成功。
二、实践练习
要求:在第一次作业的程序代码基础进行改编,除了x2x函数之外,再扩展写一个x2y函数(功能自定),main函数代码将调用x2x和x2y ;将这3个函数分别写成单独的3个 .c文件,并用gcc分别编译为3个.o 目标文件;将x2x、x2y目标文件用 ar工具生成1个 .a 静态库文件, 然后用 gcc将 main函数的目标文件与此静态库文件进行链接,生成最终的可执行程序,记录文件的大小。
1、准备工作
具体文件请访问仓库地址
增加sub2.c文件,包含x2y函数
#include "sub2.h"
#include <stdio.h>
void x2y(void)
{
printf ("Hello GCC World\n");
}
增加sub2.h头文件
#ifndef _SUB2_H_
#define _SUB2_H_
void x2y();
#endif
2、静态库编译
生成.o的目标文件gcc -c sub1.c sub2.c
利用.o的目标文件,生成静态库文件 ar crv libsubfile.a sub1.o sub2.o
使用静态库文件libafile.a 编译生成程序gcc -o main1 main1.c libsubfile.a
使用size命令查看编译生成的目标大小
3、动态库编译
生成对象文件(添加命令"-fpic",否则在生成.so文件时会出错)gcc -c -fpic sub1.c sub2.c
生成动态库.so文件 gcc -shared *.o -o libsubfile.so
使用.so文件,创建可执行程序gcc -o main2 main1.c libsubfile.so,执行./main2发现报错
使用ldd命令查看,发现找不到对应的libsubfile.so文件,报错信息如下。执行拷贝命令sudo cp libsubfile.so /usr/lib,再次执行目标文件可以成功。
使用size命令查看目标文件大小。通过比较可知静态库生成的文件略大于动态库生成的文件,可能是因为.c文件太小,对比不够明显