Linux动静态库

目录

静态库和动态库概念

编写一个静态库和动态库

静态库的使用 

动态库的使用 

使用动静态库以及为什么要有库


静态库和动态库概念

  • 1.静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再要静态库
  • 2.动态库(.so):程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。
  • 3.一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在标文件的整个机器码
  • 4.在可执行文件开始运行以前,外部函数的机器码由操作系统从磁盘上的该动态库中复制到内存中,这个过程称为动态链接(dynamic linking)
  • 5.动态库可以在多个程序间共享,所以动态链接使得可执行文件更小,节省了磁盘空间。操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间。

编写一个静态库和动态库

//Makefile
.PHONY:all
all:libhello.so libhello.a

libhello.so:mymath_d.o myprint_d.o
	gcc -shared mymath_d.o myprint_d.o -o libhello.so
mymath_d.o:mymath.c
	gcc -c -fPIC mymath.c -o mymath_d.o
myprint_d.o:myprint.c
	gcc -c -fPIC myprint.c -o myprint_d.o

libhello.a: mymath.o myprint.o
	ar -rc libhello.a mymath.o myprint.o
mymath.o:mymath.c
	gcc -c mymath.c -o mymath.o
myprint.o:myprint.c
	gcc -c myprint.c -o myprint.o

.PHONY:output
output:
	mkdir -p output/lib
	mkdir -p output/include
	cp -rf *.h output/include
	cp -rf *.a output/lib
	cp -rf *.so output/lib

.PHONY:clean
clean:
	rm -rf *.o *.a *.so output
//myprint.h
#pragma once

#include <stdio.h>
#include <time.h>

extern void Print(const char *str);
//myprint.c
#include "myprint.h"

void Print(const char *str)
{
    printf("%s[%d]\n", str, (int)time(NULL));
}
//mymath.h
#pragma once

#include<stdio.h>

extern int addToTarget(int from, int to);
//mymath.c
#include "mymath.h"


int addToTarget(int from, int to)
{
    int sum = 0;
    for(int i = from; i <= to; i++)
    {
        sum += i;
    }

    return sum;
}

 

第一步的准备工作

然后我们的静态库和动态库就出来了,我们想要进行编译的话,是需要.o和.h文件的,但是我们有了库之后就不需要.o文件了,因为都到库里了,我们就需要去另一个目录该他们包装一下.

把他打包成这样就可以了

静态库的使用 

第一种方法是拷贝到系统目录下

头文件gcc的默认搜索路径是:/usr/include                                                                                        库文件的默认搜索路径是:/lib64 or /usr/lib64 

指令:sudo cp  output/include/* /use/include/ -rf                                                                        sudo cp -rf output/lib/libhello.a /lib64                                                                                               使用静态库会需要链接,-l后面接库名去前缀lib去后缀.a编译即可:gcc main.c -lhello

第二种方法

这个是因为原本他是会在当前路径下和C语言默认的路径下搜索,现在不在这两个中,就需要自己指定。

动态库的使用 

 第一种方法和静态库一样加载到系统的目录下,就不多说了

第二种,加入到环境变量,因为他在找的时候,会在系统默认的路径下找,,和在环境变量中找,添加环境变量,这样添加的话,如果终端一关闭的话,就会消失,因为这是内存级别的添加

指令:export LD_LIBRARY_PATH:/home/chx/2022/12/28/uselib/output/lib

第三种更新配置文件:这样是永久性的,这个其实就是先在/etc/ld.so.conf.d这个路径下创建一个文件,然后把你要加的那个库的路径写进去即可最后执行一下sudo ldconfig更新一下配置文件即可

指令:添加:sudo touch /etc/ld.so.conf.d/hello.conf                                                                      查看:ls /etc/ld.so.conf.d -l                                                                                                              写入路径:sudo vim /etc/ld.so.conf.d/hello.conf                                                                             刷新:sudo ldconfig

第四种在系统路径下建立一个软链接

指令:sudo ln -s /home/chx/2022/12/28/uselib/output/lib/libhello.so /lib64/libhello.so

最后可以用(ldd 可执行文件)查看链接情况。

使用动静态库以及为什么要有库

如果同时有静态库和动态库,默认使用动态库, 站在是一个使用者(程序员)角度-- 使用以动态库,a.如果我们只有静态库,没办法,gcc只能针对该库进行静态链接                                                   b.如果动静态库同时存在,默认用的就是动态库                                                                              c.如果动静态库同时存在,我非要使用静态库呢?指令后面加static,static的意义:摒弃默认优先使用动态库的原则,而是直接使用静态库的方案                                                                              d.在使用动态库时,在执行可执行文件的时候,也要告诉他,,因为动态库是动态链接的,静态库是程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再要静态库。形成了可执行程序为什么还是执行出错,因为我的可执行程序并不知道,所以我们要用别的方法。

注意:动态库是一个独立的库文件,动态库可以和可执行程序,分批加载。

为什么要有库,站在使用库的角度,库的存在,可以大大减少我们开发的周期,提高软件本身的质量站在写库的人的角度,1.简单 2.代码安全。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

pythoncjavac++

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值