动静态库(Linux)


前言

我们之前用过c语言的库.Linux中默认的都是使用动态库,如果想要使用静态库,就必须加上-static选项。默认都是安装的动态库,系统中一般没有静态库,如果要使用,就需要自己安装。
Linux下动态库是以.so结尾的,静态库是以.a结尾的。
无论是动态库还是静态库,都有自己的名字。比如libc.so,库的名字需要去掉前缀lib,去掉后缀.so,那么这就是一个c标准库。

一、静态库

什么是库呢??
库本质就是把一些.o结合放在一起就是我们使用的库

我们只要拥有.o文件和.h文件就可以调用相关的函数。 比如下面的例子

add.h

#pragma once
int add(int x,int y);

add.c


#include "add.h"
int add(int x,int y)
{
    return x+y;
}

myfile.c


#include "myfile.h"

//打开
myFILE*myopen(const char*path,const char*mode)
{
    int flag=0;
    if(strcmp(mode,"w")==0)
    {
        flag|=O_CREAT|O_WRONLY|O_TRUNC;
    }
    else if(strcmp(mode,"a")==0)
    {
        flag|=O_CREAT|O_APPEND|O_TRUNC;
    }
    else if(strcmp(mode,"r")==0)
    {
        flag|=O_RDONLY;
    }
    else
    {
        return NULL;
    }
    int fd=0;
    if(flag&O_RDONLY)
    {
        fd=open(path,flag);
    }
    else
    {
        umask(0);
        fd=open(path,flag,0664);
    }

    myFILE*fp=(myFILE*)malloc(sizeof(myFILE));
    fp->inode=fd;
    fp->pos=0;
    fp->cap=MAX;
    fp->flushmode=FLUSHLINE;
    return fp;

}
void myfflush(myFILE*fp)
{
    write(fp->inode,fp->buffer,fp->pos);
    fp->pos=0;
}
//写文件
int myfwrite(const void *ptr,size_t n,myFILE*fp)
{
    //首先放到缓冲区
    memcpy(fp->buffer+fp->pos,ptr,n);
    fp->pos+=n;

    //判断是否需要刷新
    if(fp->flushmode==1&&fp->buffer[fp->pos-1]=='\n')
    {
        myfflush(fp);
    }
    if(fp->flushmode==2&&fp->pos==fp->cap)
    {
        myfflush(fp);
    }
    return n;
}
//关闭
void  myclose(myFILE*fp)
{
    //进程退出先刷新
    myfflush(fp);
    close(fp->inode);
     free(fp);
}


myfile.h

#pragma once 

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>

#define FLUSHLINE 1
#define FLUSHFULL 2
#define FLUSHNO 3
#define MAX 4096

typedef struct myFILE
{
    int inode;
    char buffer[MAX];
    int pos;
    int cap;
    int flushmode;

}myFILE;

//打开
myFILE*myopen(const char*path,const char*mode);
//写文件
int myfwrite(const void *ptr,size_t n,myFILE*fp);
//刷新
void myfflush(myFILE*fp);
//关闭
void myclose(myFILE*fp);

测试代码test.c

#include <stdio.h>
#include "add.h"
#include "myfile.h"

int main()
{
    int x=10;
    int y=20;
    int sum=add(x,y);
    printf("%d+%d=%d\n",x,y,sum);

    myFILE*fp=myopen("file.txt","w");
    if(fp==NULL) 
    {
        perror("myopen:");
        return 1;
    }
    
    int cnt=5;
    char buffer[64]={"hello world\n"};
    while(cnt--)
    {
        myfwrite(buffer,64,fp);
    }
    myclose(fp);

    
    return 0;
}

1.正常测试
下面是我们的结构,我们自己写了两个add和myfile两个函数文件1,我们并不像把.c文件给user,但是好像让他用我们的函数,我们就可以把.o和.h给他,然后这个用户自己写main函数就可以。
在这里插入图片描述
我们测试一下效果
在这里插入图片描述
我们运行看一下结果
在这里插入图片描述
确实完成了我们的任务。
2.打包测试
我们同样也可以把这些.o文件进行打包操作,形成一个文件
在静态库下我们用的打包操作是:ar -rc ,注意按照库的格式命名
查看打包库的情况:ar -tv

在这里插入图片描述
把打包的库文件和.h拷贝到该用户下,编译运行test.c
在这里插入图片描述
我们后发现报错了???这是为什么呢???
我们自己安装的属于第三方库,系统默认回去自己默认的路径去寻找,就不会找到这个库文件,虽然他们在同一路径下。

我们需要指明库的路径和库名称,采用下面指令
gcc 文件 -L(跟路径) -l(跟库名称)
在这里插入图片描述
为什么要有库呢??测试目标文件生成之后,静态库就可以删掉了,程序照样可以正常运行

为什么要有库呢??
1.提高开发效率
2.隐藏源代码

二、动态库

同样还是上面的代码,我们继续测试动态库

1.正常测试
我们如果想要对动态库进行编辑,需要带上 -fPIC(产生位置无关码)选项
在这里插入图片描述
我们如果想对这些.o文件进行打包,我们需要借助-shared(动态格式)选项,对动态库打包不需要其他选项,直接使用gcc就可以
在这里插入图片描述
此时我们就可以正常对test.c进行编译。但是会报错
在这里插入图片描述
此时就与静态库相同,动态库也属于第三方库,需要指定库的路径和库名称
在这里插入图片描述
我们平时使用的库都是把.o文件和.h文件进行打包,上传到网络中,使用者将库下载下来,进阶捷报就可以正常操作了。

下面我们模拟一下这个过程
makefie中内容


libmyc.so:add.o myfile.o
	gcc -shared -o $@ $^
%.o:%.c
	gcc -c -fPIC $<

.PHONY:clean
clean:
	rm -rf *.o libmyc.so  mylib mylib.tgz

.PHONY:output
output:
	mkdir -p mylib/include
	mkdir -p mylib/lib
	cp -rf *.h mylib/include
	cp -rf *.so mylib/lib
	tar czf mylib.tgz mylib

2.安装测试
我们进行make output操作将相应的内容进行打包
在这里插入图片描述
用户把这个包下载下来,并且解压到自己的路径下,之后把这个压缩包删除就可以。
所谓的把库安装到系统中,本质就是把对应的文件拷贝到系统指定的路径中。
在这里插入图片描述
下面我们再次进行编译,我们可以发现报了一个这样的错误
在这里插入图片描述
头文件找不到,Linux下查找头文件会在当前路径和库对应的路径进行查找,我们的.h文件不在在下级路径下,系统自然就找不到。
我们需要用到-I(大写i)指定头文件路径,同时用-L(指定库路径)和-l(指定库名称)
在这里插入图片描述

3.运行测试
我们正常运行看一下
在这里插入图片描述
发现报错了??这是为什么呢??
编译时我们确定了编译环境所需要的路径以及一系列东西,但是对于运行程序的进程来说,并不知道库的路径(第三方库)。

我们也可以用ldd查看链接了哪些库

在这里插入图片描述

我们要解决这个有四种方法 🌟直接把所使用的库拷贝到系统指定的路径下,一般是/usr/lib(sudo提权)
在这里插入图片描述
🌟使用环境变量LD_LINRARY_PATH,这个环境变量是指,操作系统查找库文件时候,除了去系统指定的目录去查找,同时也会去这个路径下查找
在这里插入图片描述
但是如果我们关掉机器,重新打开,这个环境变量就会消失。如果想让他一直存在,可以在跟目录下的_bash_profile和.bashrc修改即可

🌟使用软链接的方式
在这里插入图片描述
🌟设置配置文件/etc/ld.so.conf.d
将库的路径放在这个目录中,目录名称必须以.conf结尾,之后使用ldconfig使配置生效
在这里插入图片描述

4.动态库&&静态库
🌟如果动态库和静态库同时存在,系统自动使用动态库,如果非要使用静态库,加-static选项
🌟如果我们所用的文件只存在静态库,系统将把这个链接静态库,其余正常链接静态库
🌟如果我们只提供动态库,非得想要静态链接,会报错

三、深入理解动态库

1.系统角度


运行代码,首先创建task_struct ,进程地址空间,将磁盘中test.c文件加载进内存,填写页表内容。动静态库在磁盘中也是一个个的文件,也要加载到内存,放在页表中,与物理内存建立映射。
🌟静态库:把test.c和libmyc.o一起加载到内存中,放在一起。运行时就不需要寻找库了。
🌟动态库:把test.c和libmyc.sof分别加载到内存中,其中动态库会被放在进程地址空间的共享区中。当调用这个库时,去共享区中查找,如果存在这个库,就跳转到共享库中调用。如果不存在就加载到内存中。如果其他文件要调用这个动态库,就直接去调用即可,不用再进行加载。动态库也叫做共享库,本质就是进程中代码和数据仅存在一份。

补充: 1.进程地址空间的内容初始化是根据文件的代码数据进行填充的。
   2.进程同样要对这些库进行管理,怎么管理呢??先描述,在组织
   3.那些库加载了,那些库没有被加载,由操作系统自动完成。
2.编译角度
在磁盘中的文件中代码和数据在没有加载进入内存之前,也有自己对应的地址,数据和代码按照一定的格式进行存储。我们就可以用这个初始化我们进程的进程地址空间了。
编址方式有两种:绝对地址和相对地址,机器一般使用绝对编制的方式。

总结

以上就是今天要讲的内容,本文仅仅详细介绍了 。希望对大家的学习有所帮助,仅供参考 如有错误请大佬指点我会尽快去改正 欢迎大家来评论~~ 😘 😘 😘

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

lim 鹏哥

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

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

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

打赏作者

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

抵扣说明:

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

余额充值