Linux:软硬链接和动静态库

在这里插入图片描述
hello,各位小伙伴,本篇文章跟大家一起学习《Linux:软硬链接和动静态库》,感谢大家对我上一篇的支持,如有什么问题,还请多多指教 !
如果本篇文章对你有帮助,还请各位点点赞!!!

话不多说,开始正题:

文章目录

    • 创建软硬链接
      • 理解软硬链接
      • 为什么要有软硬链接?
    • 静态库
      • 方法一:安装到系统里
      • 方法二:和源文件在一起
      • 方法三:使用带路径的库
    • 动态库
      • 方法一:把动态库安装到系统中
      • 方法二:和源文件在一起
      • 方法三:使用带路径的库
      • 解决问题
    • 同时提供动静态库,系统会选择哪一个?
    • 原理上理解动态库

创建软硬链接

软连接:

ln -s 目标文件 软连接文件

在这里插入图片描述
看到file-sort.link有自己的inode,这就说明file-sort.link是一个独立的文件,向file.txt里写入东西:
在这里插入图片描述
发现可以从软链接中获取file.txt的内容,接下来创建硬链接:

ln 目标文件 软连接文件

在这里插入图片描述
发现硬链接的inode和目标文件的一样,说明硬链接不是一个独立的文件,发现目标文件的数字1变成了2,要是把硬链接给删掉:
在这里插入图片描述
又变回数字1

理解软硬链接

  • 软链接实际上就是保存你目标文件的路径,就像我们Windows的快捷方式,点击桌面的快捷方式就可以打开文件
  • 硬链接本质就是一组文件名和已经存在的文件的映射关系,可以理解为给文件多取了一个新的名字
  • 为什么数字会变成2?其实这里的数字表示的是映射关系的个数,在这里有两个文件名指向目标文件,所以数字就会变成2,但要是把硬链接给删除,数字就变回了1
  • 实际上就是inode里维护着一个引用计数(也叫做硬链接数)记录着有多少个文件名指向着文件,所以要把一个文件真正的删除,就要把所有指向该文件的文件名全部删除,也就是引用计数为0
  • 要是把原先文件名给删掉,那不就相当于给我文件进行重命名了吗?但是要注意,这时候的软链接就失效了,因为该软链接找不到文件名了
    在这里插入图片描述

为什么要有软硬链接?

软链接:
可以直接将可执行程序通过软链接直接链接到/usr/bin/目录底下,这样就可以直接执行可执行程序不需要带路径了
在这里插入图片描述
在这里插入图片描述
想要删掉软链接也可以使用unlink
在这里插入图片描述
删除后再运行code就找不到了

也可以对目录进行软链接,比如有些目录路径太深了,就可以使用软链接到当前目录,所以软链接最主要的作用就是方便操作

对于硬链接,创建一个空目录,这个目录默认就有两个链接数:
在这里插入图片描述
这是因为对于这个目录有两个文件名指向它,一个是empty,另一个是.,没错这个.就是表示当前目录的意思,所以数字是2

当我在该空目录底下创建一个新目录dir
在这里插入图片描述
就会发现链接数变成了3,那是因为新目录dir..来指向上级目录,所以多了一个链接数
在这里插入图片描述
但是根目录有点特殊,因为是被特殊处理过的,当在根目录想继续切换到上级目录:

cd ..

是停留在根目录的,因为根目录这个..是特殊处理过的,也不影响链接数,也删不掉

硬链接可以对大文件进行备份并且不需要进行拷贝,节省空间的拷贝

但是,Linux中不允许对目录新建硬链接:
在这里插入图片描述
因为会形成环形路径:
在这里插入图片描述
软链接就不会,因为系统不会对软链接进行解析,读到软链接就不继续操作了,但是操作系统就可以创建目录的硬链接...嘛,你别管,操作系统就是老大

静态库

方法一:安装到系统里

先创建一些头文件和.o文件来创建静态库:
my_stdio.c:

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

mFILE *mfopen(const char *filename, const char *mode)
{
    int fd = -1;
    if(strcmp(mode, "r") == 0)
    {
        fd = open(filename, O_RDONLY);
    }
    else if(strcmp(mode, "w")== 0)
    {
        fd = open(filename, O_CREAT|O_WRONLY|O_TRUNC, 0666);
    }
    else if(strcmp(mode, "a") == 0)
    {
        fd = open(filename, O_CREAT|O_WRONLY|O_APPEND, 0666);
    }
    if(fd < 0) return NULL;
    mFILE *mf = (mFILE*)malloc(sizeof(mFILE));
    if(!mf) 
    {
        close(fd);
        return NULL;
    }

    mf->fileno = fd;
    mf->flag = FLUSH_LINE;
    mf->size = 0;
    mf->cap = SIZE;

    return mf;
}

void mfflush(mFILE *stream)
{
    if(stream->size > 0)
    {
        // 写到内核文件的文件缓冲区中!
        write(stream->fileno, stream->outbuffer, stream->size);
        // 刷新到外设
        fsync(stream->fileno);
        stream->size = 0;
    }
}

int mfwrite(const void *ptr, int num, mFILE *stream)
{
    // 1. 拷贝
    memcpy(stream->outbuffer+stream->size, ptr, num);
    stream->size += num;

    // 2. 检测是否要刷新
    if(stream->flag == FLUSH_LINE && stream->size > 0 && stream->outbuffer[stream->size-1]== '\n')
    {
        mfflush(stream);
    }
    return num;
}

void mfclose(mFILE *stream)
{
    if(stream->size > 0)
    {
        mfflush(stream);
    }
    close(stream->fileno);
}


my_stdio.h:

#pragma once

#define SIZE 1024

#define FLUSH_NONE 0
#define FLUSH_LINE 1
#define FLUSH_FULL 2

struct IO_FILE
{
    int flag; // 刷新方式
    int fileno; // 文件描述符
    char outbuffer[SIZE];
    int cap;
    int size;
    // TODO
};

typedef struct IO_FILE mFILE;

mFILE *mfopen(const char *filename, const char *mode);
int mfwrite(const void *ptr, int num, mFILE *stream);
void mfflush(mFILE *stream);
void mfclose(mFILE *stream);

my_strinng.c:

#include "my_string.h"

int my_strlen(const char *s)
{
    const char *end = s;
    while(*end != '\0')end++;
    return end - s;
}

my_string.h:

#pragma once

int my_strlen(const char *s);

.c文件进行编译:

gcc -c *.c

创建静态库:

ar -rc lib库名字.a.版本号

必须是以lib开头.a结尾,后面的版本号可以写可以不写:
在这里插入图片描述

将头文件拷贝到系统的头文件里,将新建的静态库拷贝到系统的静态库里:
在这里插入图片描述

接下来在另一个目录里面尝试使用我们所导入的静态库:

#include <my_stdio.h>
#include <my_string.h>
#include <stdio.h>

int main()
{
    char *str = "hello";
    int len = my_strlen(str);
    printf("len: %d\n", len);
    mFILE *fp = mfopen("log.txt", "a");
    if(fp == NULL) return 1;
    
    mfwrite(str, len, fp);
    mfclose(fp);
    return 0;
}

在这里插入图片描述
发现链接报错!报错信息显示并不认识所用的函数,这是为什么?

首先,在lib64/里面并不只有一个库,编译器怎么知道你用的是哪一个库

这不对吧,那为什么我们平时用gcc的时候并不需要告诉编译器要用哪个库呢?
那是因为gcc是C语言编译,肯定是要认识C标准库的啊

所以我们要告诉编译器要用哪个库,无论你怎么操作,你都只是第三方库:

gcc test.c -llibmystdio.a

在这里插入图片描述
怎么还是不行?我已经告诉他了啊,那是因为告诉错了,库的名字是要去掉前缀和后缀:

gcc test.c -lmystdio

在这里插入图片描述

方法二:和源文件在一起

懂了吗?要是在目录中已经提供了.h静态库该怎那么使用呢?首先就是吧头文件改回""

#include "my_stdio.h"
#include "my_string.h"
#include <stdio.h>

int main()
{
    char *str = "hello";
    int len = my_strlen(str);
    printf("len: %d\n", len);
    mFILE *fp = mfopen("log.txt", "a");
    if(fp == NULL) return 1;

    mfwrite(str, len, fp);
    mfclose(fp);
    return 0;
}

在这里插入图片描述

gcc test.c -lmystdio

在这里插入图片描述

竟然发现找不到库?明明已经在当前目录了啊,这就说明编译器在查找库的时候,并不会在当前目录下查找,所以引入另一个选项-L

gcc test.c -L. -lmystdio

-L之后指明路径,告诉编译器不仅要在系统库里查找,还要在我知名的路径下查找,在这里我写的是.,表示当前目录下
在这里插入图片描述

方法三:使用带路径的库

Makefile:

libmystdio.a:my_stdio.o my_string.o
	ar -rc $@ $^
%.o:%.c
	gcc -c $<
.PHONY:clean
clean:
	rm -rf *.a *.o output
.PHONY:output
output:
	mkdir -p stdc/include
	mkdir -p stdc/lib
	cp -f *.h stdc/include
	cp -f *.a stdc/lib

把弄好的库和头文件全部打包在stdc里:
在这里插入图片描述

对于这种情况,我们该怎么使用库?其实很简单,首先就是告诉编译器头文件在哪里:

-I std/include //可以有空格
-Istd/include //也可以无空格

然后就是告诉编译器库在哪里:

-L std/lib //可以有空格
-Lstd/lib //也可以无空格

最后就是告诉编译器哪一个库:

-lmystdio

总结:

gcc -o test test.c -Istdc/include -Lstdc/lib -lmystdio

在这里插入图片描述

动态库

怎么形成动态库:

gcc -o lib动态库.so 依赖文件 -shared // -shared就是告诉编译器不是形成可执行文件,是形成动态库

Makefile

libmystdio.so:my_stdio.o my_string.o
	gcc -o $@ $^ -shared
%.o:%.c
	gcc -fPIC -c $<
.PHONY:clean
clean:
	rm -rf *.so *.o stdc*

对于动态库,在.c文件编译成.o文件时要多加一个选项-fPIC,意思是形成与位置无关码(后续讲动态库链接时解释

方法一:把动态库安装到系统中

导入库中:

 sudo cp *.so /usr/lib/

我们可以通过ldd指令来查看该可执行程序所依赖的库:

ldd ./a.out

在这里插入图片描述
即使形成了可执行文件,动态库也不能删除,删了就跑不动了
在这里插入图片描述

方法二:和源文件在一起

操作和静态库一样,但是
在这里插入图片描述
他找不到?先继续看方法三:

方法三:使用带路径的库

和静态库方法一样,但是
在这里插入图片描述
还是找不到?

解决问题

第一个方法:在Ubuntu下,系统查找动态库是在/lib下查找的,所以要让我们的动态库加载到/lib目录下,这个工作很简单,可以直接拷贝:

sudo cp ./libmystdio.so /lib/

在这里插入图片描述

第二个方法:直接在/lib目录下创建软链接与我们的动态库进行链接就行了:

sudo ln -s /home/Sherry/linux_learn/23_day/so/libmystdio.so /lib/libmystdio.so

一定要注意:软链接一定要与动态库同名
在这里插入图片描述
此时就可以执行了

第三个方法:导入环境变量

有没有想过为什么系统自身的动态库不需要进行这么复杂的操作?那是因为系统会默认去环境变量查找动态库,这个环境变量叫做:

LD_LIBRARY_PATH

一般没有配置过文件的是没有的,我们可以自行导入环境变量:

export LD_LIBRARY_PATH=LD_LIBRARY_PATH:/home/Sherry/linux_learn/23_day/so

在这里插入图片描述

同时提供动静态库,系统会选择哪一个?

在这里插入图片描述
同时提供动静态库,系统会默认选择动态库,若要想使用静态库,就要带上:

-static

要注意:如果你强制要使用静态链接,那么就必须提供静态库!
要是你没有提供动态库,只有静态库,用了动态链接,那么gcc、g++都没得选,只能针对你提供的静态库局部性采用静态链接,程序也可以跑

原理上理解动态库

动态库其实也是个文件,存放在磁盘中,要是有程序使用到这个动态库,就要将动态库加载到内存里

一个程序对应一个主进程,在进程里有对应的虚拟内存,虚拟内存依赖页表来与内存进行映射关系,当磁盘中的动态库加载到内存中后,虚拟内存里的共享区就会通过页表产生映射关系,这样程序就可以找到动态库

当然,一个动态库并非只服务于一个程序,可以是上百个程序,然而动态库已经加载到内存里,所以,只需要虚拟内存通过页表产生映射关系即可,无需再次加载动态库,大大节省了空间,也就是为什么,动态库也被称为共享库,也就是为什么,优先使用动态库

你学会了吗?
好啦,本章对于《Linux:软硬链接和动静态库》的学习就先到这里,如果有什么问题,还请指教指教,希望本篇文章能够对你有所帮助,我们下一篇见!!!

如你喜欢,点点赞就是对我的支持,感谢感谢!!!

请添加图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/967617.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

CSS 组合选择符详解与实战示例

在 Web 开发过程中&#xff0c;CSS 用于定义页面元素的样式&#xff0c;而选择器则帮助我们精确定位需要添加样式的元素。今天我们主要来讲解 CSS 中的组合选择符&#xff0c;它们能够根据 DOM 结构中元素之间的关系来选中目标元素&#xff0c;从而写出结构清晰、易于维护的 CS…

【Linux系统】—— 简易进度条的实现

【Linux系统】—— 简易进度条的实现 1 回车和换行2 缓冲区3 进度条的准备代码4 第一版进度条5 第二版进度条 1 回车和换行 先问大家一个问题&#xff1a;回车换行是什么&#xff0c;或者说回车和换行是同一个概念吗&#xff1f;   可能大家对回车换行有一定的误解&#xff0…

Winform开发框架(蝇量级) MiniFramework V2.1

C/S框架网与2022年发布的一款蝇量级开发框架&#xff0c;适用于开发Windows桌面软件、数据管理应用系统、软件工具等轻量级软件&#xff0c;如&#xff1a;PLC上位机软件、数据采集与分析软件、或企业管理软件&#xff0c;进销存等。适合个人开发者快速搭建软件项目。 适用开发…

win10 llamafactory模型微调相关②

微调 使用微调神器LLaMA-Factory轻松改变大语言模型的自我认知_llamafactory 自我认知-CSDN博客 【大模型微调】使用Llama Factory实现中文llama3微调_哔哩哔哩_bilibili 样本数据集 &#xff08;数据集管理脚本处需更改&#xff0c;见报错解决参考1&#xff09; 自我认知微…

AI大模型随机初始化权重并打印网络结构方法(以Deepseekv3为例,单机可跑)

背景 当前大模型的权重加载和调用&#xff0c;主要是通过在HuggingFace官网下载并使用transformer的库来加以实现&#xff1b;其中大模型的权重文件较大&#xff08;部分>100GB&#xff09;&#xff0c;若只是快速研究网络结构和数据流变化&#xff0c;则无需下载权重。本文…

前端项目打包完成后dist本地起node服务测试运行项目

1、新建文件夹 node-test 将打包dist 文件同步自定义本地服务文件夹node-test 中&#xff0c;安装依赖包。 npm install express serve-static cors 2、新创建服务文件js server.js 构建链接及端口 const express require(express); const path require(path); const co…

《语义捕捉全解析:从“我爱自然语言处理”到嵌入向量的全过程》

首先讲在前面&#xff0c;介绍一些背景 RAG&#xff08;Retrieval-Augmented Generation&#xff0c;检索增强生成&#xff09; 是一种结合了信息检索与语言生成模型的技术&#xff0c;通过从外部知识库中检索相关信息&#xff0c;并将其作为提示输入给大型语言模型&#xff…

Word中Ctrl+V粘贴报错问题

Word中CtrlV粘贴时显示“文件未找到&#xff1a;MathPage.WLL”的问题 Word的功能栏中有MathType&#xff0c;但无法使用&#xff0c;显示灰色。 解决方法如下&#xff1a; 首先找到MathType安装目录下MathPage.wll文件以及MathType Commands 2016.dotm文件&#xff0c;分别复…

Git 与 Git常用命令

Git 是一个开源的分布式版本控制系统&#xff0c;广泛用于源代码管理。与传统的集中式版本控制系统不同&#xff0c;Git 允许每个开发者在本地拥有完整的代码库副本&#xff0c;支持离线工作和高效的分支管理。每次提交时&#xff0c;Git 会对当前项目的所有文件创建一个快照&a…

构建jdk17包含maven的基础镜像

1、先拉取jdk17基础镜像 docker pull openjdk:17-jdk-alpine 2、使用jdk17基础镜像创建容器 docker run -it openjdk:17-jdk-alpine sh 或 docker run -it --name jdk17 openjdk:17-jdk-alpine sh 3、修改镜像源地址 cat /etc/apk/repositories https://mirrors.aliyun.com…

【博客之星】GIS老矣尚能饭否?WebGIS项目实战经验与成果展示

目录 一、最前面的话 二、前言 1、关于“夜郎king” 3、GIS的“老骥伏枥” 4、WebGIS的“新程启航” 三、WebGIS技术简介 1、前、后技术简介 2、系统功能架构 四、WebGIS项目应用效果 1、应急灾害 2、交通运输 3、智慧文旅 4、其它项目 五、未来与展望 1、云计算…

如何在Vue中实现事件处理

在Vue中&#xff0c;事件处理是一个核心概念&#xff0c;它让我们能够响应用户的操作&#xff0c;比如点击按钮、输入文本等。Vue提供了一个简洁而强大的方式来绑定事件和处理事件。本文将介绍如何在Vue中实现事件处理&#xff0c;覆盖事件绑定、事件修饰符以及事件处理函数等内…

elementplus 使用日期时间选择器,设置可选范围为前后大于2年且只能选择历史时间不能大于当前时间点

需求&#xff1a;时间选择器可选的时间范围进行限制&#xff0c;-2年<a<2年且a<new Date().getTime()核心&#xff1a;这里需要注意plus版没有picker-options换成disabled-date属性了&#xff0c;使用了visible-change和calendar-change属性逻辑&#xff1a;另设一个参…

【MATLAB源码-第261期】基于matlab的帝企鹅优化算法(EPO)机器人栅格路径规划,输出做短路径图和适应度曲线

操作环境&#xff1a; MATLAB 2022a 1、算法描述 帝企鹅优化算法&#xff08;Emperor Penguin Optimizer&#xff0c;简称EPO&#xff09;是一种基于自然现象的优化算法&#xff0c;灵感来自于帝企鹅在南极极寒环境中的生活习性。帝企鹅是一种群居动物&#xff0c;生活在极端…

协议-ACLLite-ffmpeg

是什么&#xff1f; FFmpeg是一个开源的多媒体处理工具包&#xff0c;它集成了多种功能&#xff0c;包括音视频的录制、转换和流式传输处理。FFmpeg由一系列的库和工具组成&#xff0c;其中最核心的是libavcodec和libavformat库。 libavcodec是一个领先的音频/视频编解码器库&…

DuckDB:pg_duckdb集成DuckDB和PostgreSQL实现高效数据分析

pg_duckdb是PostgreSQL的扩展&#xff0c;它将DuckDB的列矢量化分析引擎和特性嵌入到PostgreSQL中。本文介绍pg_duckdb插件安装、特点以及如何快速入门使用。 pg_duckdb简介 pg_duckdb扩展将完全能够查询DuckDB中存储在云中的数据&#xff0c;就像它是本地的一样。DuckDB的“双…

防火墙安全综合实验

防火墙安全综合实验 一、拓扑信息 二、需求及配置 实验步骤 需求一&#xff1a;根据下表&#xff0c;完成相关配置 设备接口VLAN接口类型SW2GE0/0/2VLAN 10AccessGE0/0/3VLAN 20AccessGE0/0/1VLAN List&#xff1a;10 20Trunk 1、创建vlan10和vlan20 2、将接口划分到对应…

Vue 响应式渲染 - 过滤应用

Vue 渐进式JavaScript 框架 基于Vue2的学习笔记 - Vue响应式渲染综合 - 过滤应用 目录 过滤应用 引入vue Vue设置 设置页面元素 模糊查询过滤实现 函数表达式实现 总结 过滤应用 综合响应式渲染做一个输入框&#xff0c;用来实现&#xff1b;搜索输入框关键词符合列表。…

一文学会:用DeepSeek R1/V3 + AnythingLLM + Ollama 打造本地化部署的个人/企业知识库,无须担心数据上传云端的泄露问题

文章目录 前言一、AnythingLLM 简介&基础应用1.主要特性2.下载与安装3.配置 LLM 提供商4.AnythingLLM 工作区&对话 二、AnythingLLM 进阶应用&#xff1a;知识增强使用三、AnythingLLM 的 API 访问四、小结1.聊天模式2.本地存储&向量数据库 前言 如果你不知道Olla…

CNN-LSTM卷积神经网络长短期记忆神经网络多变量多步预测,光伏功率预测

CNN-LSTM卷积神经网络长短期记忆神经网络多变量多步预测&#xff0c;光伏功率预测 一、引言 1.1、研究背景和意义 光伏发电作为一种清洁能源&#xff0c;对于实现能源转型和应对气候变化具有重要意义。然而&#xff0c;光伏发电的输出功率具有很强的间歇性和波动性&#xff…