Linux下软硬链接和动静态库制作详解

news2025/1/16 18:52:36

目录

前言

软硬链接

概念 

软链接的创建

硬链接的创建

软硬链接的本质区别

理解软链接

理解硬链接

小结

动静态库

概念

动静态库的制作

静态库的制作 

动态库的制作 


前言

本文涉及到inode和地址空间等相关概念,不知道的小伙伴可以先阅读以下两篇文章了解一下哦。

 Linux文件系统

Linux进程地址空间

觉得内容对你有所帮助的话可以给博主一键三连哦 

你们的支持将是我继续创作的动力

如果内容有错或者不足的话,还望能指出

软硬链接

概念 

软链接(Symbolic Link):是一个特殊的文件,它指向另一个文件或目录。

硬链接(Hard Link):是指向文件数据块的另一个文件名。

软链接的创建

命令:ln -s [源文件] [链接文件]

从inode编号可以看出软链接其实就是一个独立的文件。

硬链接的创建

命令:ln [源文件] [链接文件]

从inode编号可以看出这两个文件其实是同一个,只不过是不同的名字。

软硬链接的本质区别

软硬链接的本质区别就在于有没有独立的inode,软链接有独立的inode,所以软链接是一个独立的文件;硬链接没有独立的inode,所以硬链接不是一个独立的文件。可以这么理解软链接就相当于你在windows下给一个应用创建了一个快捷方式;硬链接就相当于你将一个文件拷贝了一下并且改了一下文件名。

理解软链接

软链接创建时会被占用一定的磁盘空间,它只是一个指向原始文件或目录的指针,如果原始文件或目录被删除,软链接仍然存在,但无法访问到真正的文件或目录。

理解硬链接

创建硬链接没有独立的inode,所以不是真正的创建了一个新文件,所以硬链接不占用空间。实际上创建硬链接就是在指定的目录下建立了不同文件名和指定inode的映射关系仅此而已。

再来看看下面的实验

在创建硬链接后,硬链接数由1变为了2

删除文件后,硬链接数由2变为了1

也可以用unlink进行删除。 

如何理解这里的硬链接数呢?

其实这个硬链接数就是一个引用计数和C++智能指针中的shared_ptr差不多,当创建一个硬链接时,硬链接数会+1,删除文件时也只是在目录中将文件名和inode的映射关系删掉,并且把硬链接数-1,而inode还在并没有被删掉,只有当硬链接数减为0时才是真正意义上的将文件删掉了。

那么我们创建一个目录看一下硬链接数又会有什么现象?

为什么这个默认创建的目录,硬链接数是2呢?

在默认创建的目录下会有两个默认的文件.和.. 

默认创建的目录硬链接数是2,是因为:自己目录名和inode建立了映射关系,自己目录下的.和inode也建立了关系,所以这个.表示的是当前目录(所表示的就是tmp,只不过名字不同罢了)。

在这个tmp目录下,再创建一个目录硬链接数又会发生什么变化呢?

硬链接数由2变为了3 

从上图中可以看到dir目录下的..的inode编号是不是和tmp的inode编号一样,所以硬链接数由2又变为了3的原因是在dir目录下的..和指定的tmp的inode也建立了映射关系,而..代表的就是上级目录(所表示的也是tmp,只不过名字不同罢了),所以你在 cd ..时会返回到上级目录。

小结

  • 硬链接就是同一文件的不同文件名,创建硬链接就是在指定目录下建立了不同文件名和指定inode的映射关系仅此而已。
  • 软链接是一个独立的文件,相当与创建了一个快捷方式。
  • 在默认创建的目录下会有两个默认的文件.和..,.代表的就是当前目录,..代表的就是上级目录。
  • 硬链接数本质是一个引用计数。

动静态库

概念

静态库(.a结尾):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库

动态库(.so结尾):程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。

动态链接

一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在目标文件的整个机器码,在可执行文件开始运行以前,外部函数的机器码有操作系统从磁盘上的该动态库中复制道内存中,这个过程称为动态链接(dynamic linking)。动态库可以在多个程序间共享,所以动态链接使得可执行文件更小,节省了磁盘空间。操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间。

静态链接

静态链接是一种将程序所需的所有库函数和外部模块都在编译时候链接到最终的可执行文件中的链接方式。在静态链接中,编译器将程序中所有用到的库函数和外部模块的代码复制一份到最终的可执行文件中,使得最终的可执行文件可以独立运行,不依赖于系统中是否存在相应的库函数和外部模块。

动静态库的制作

这里就写了两个函数——一个累加函数和一个打印函数,用来演示动静态库的制作

累加函数 

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;
}

打印函数

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));
}

静态库的制作 

制作的静态库要能被对方使用必须得将上面两个函数编译生成.o文件(目标文件),之后将编译好的.o文件和写好的.h文件打包给别人,别人就能使用了。

main.c

#include "myprint.h"
#include "math.h"

int main()
{
    Print("hello world");
    int res = addToTarget(1, 100);

    printf("res: %d\n", res);

    return 0;
}

将上面生成的两个.o文件和main.o文件共同链接形成可执行程序 

 

虽然说这样的gcc写法是可以的,但是如果文件一多,编译的时候要把那么多的.o文件都写上去实在是太麻烦了。所以我们可以使用命令ar -rc将两个函数生成的.o文件进行归档。

ar(archive 归档)是gnu归档工具,rc表示的是(replace and create)

制作静态库

libhjx.a就是将mymath.o和myprint.o归档后生成的静态库,前面的lib和.a后缀是必带的,而hjx是这个静态库的名字(下面的动态库也是一样的)。所以使用ar -rc归档的过程就是在制作静态库。

查看静态库中的目录列表

t:列出静态库中的文件

v:详细信息

但是我们还想让它自动化的帮我们生成.o文件和静态库,那就需要使用makefile了。

编写makefile

libhjx.a:mymath.o myprint.o
	ar -rc libhjx.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:clean
clean:
	rm -rf *.o libhjx.a

我们要怎么将我们制作好的静态库给别人使用呢?

通常情况下,我们会弄一个目录,比如就叫做hjx,目录中包含一个include目录和一个lib目录,include目录下都是库的所有头文件,而lib目录下是对应的静态库文件。最后将这个目录打包给对方使用

因此我们编写的makefile还要添加自动化打包功能

libhjx.a:mymath.o myprint.o
	ar -rc libhjx.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:hjx 
hjx:
	mkdir -p hjx/lib 
	mkdir -p hjx/include
	cp -rf *.h hjx/include
	cp -rf *.a hjx/lib 

.PHONY:clean
clean:
	rm -rf *.o libhjx.a hjx

这样就不需要敲太多指令了,全部交给makefile自动帮我们生成了。 

对方拿到了我们的这个hjx目录之后要怎么使用呢?

方法一:将我们的静态库拷贝到系统路径中 

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

库文件的默认搜索路径是:/lib64 或者 /usr/lib64

所以我们要把我们的头文件拷贝到/usr/include下,库文件拷贝到/lib64下

  • 拷贝头文件

  • 拷贝库文件

在编译的时候我们要带上静态库名来表示我们要链接哪个静态库

-l:指定库名

而我们上面拷贝头文件和库文件到系统的默认路径下就叫做库的安装

但是这种方法还是不可取的,因为你写的库是没有做任何可靠性验证测试的,而系统的库都是经过大佬们精心打磨过的,你把你写的库拷贝到系统中可能就会造成库的污染。

方法二:手动指定静态库路径和头文件路径

gcc main.c -I ./hjx/include/ -L ./hjx/lib/ -lhjx

-I:指定路径搜索头文件

-L:指定库路径

注:一定要加上库名因为你lib目录下的库文件如果有很多,gcc就不知道你究竟要链接哪个库了,所以gcc要求带上库名指定你要链接哪个库。

动态库的制作 

和静态库制作那里一样,我们也是要生成.o文件但是在使用gcc生成的时候要加-fPIC选项表示生成位置无关代码

然后再用这里的.o文件来制作我们的动态库。在终端输入以下命令

gcc -shared myprint_d.o mymath_d.o -o libhjx.so
 

这里的-shared选项表示的是生成动态库

同样的这样让我们手动去将文件一个个写上去太麻烦了,所以我们需要重新编写makefile,让它自动化生成。

.PHONY:all
all:libhjx.so libhjx.a

libhjx.so:mymath_d.o myprint_d.o
	gcc -shared mymath_d.o myprint_d.o -o libhjx.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

libhjx.a:mymath.o myprint.o
	ar -rc libhjx.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

使用动态库

可以看到output文件中是我们制作的动态库

我们和静态库一样,手动指定库路径和头文件路径,但是静态库和动态库的名字都是一样的,那么gcc会使用哪一个库呢?

可以看到当我们运行生成的可执行程序时报错了,报错的内容是不能打开动态库,并且使用ldd来查看链接的动态库时法相libhjx.so没有找。但是我们可以得出结论gcc默认优先使用的是动态库,既然gcc默认优先使用的是动态库,那么我们想使用静态库怎么办,可以加-static选项。

 gcc main.c -I ./output/include/ -L ./output/lib/ -lhjx -static

-static的意义摒弃默认优先使用动态库的原则,而是直接使用静态库的方案。 

可是话说回来为什么上面会报错呢,output里面明明就有动态库啊?

首先要了解我们的可执行程序和动态库是分批加载的,可执行程序加载时,是先到地址空间中的代码区,然后通过页表映射到内存中,当我们的动态库加载时,先到内存中,然后通过页表映射到地址空间中的共享区(共享区存放的是库的起始加载虚拟地址+函数的偏移量),之后操作系统就可以通过共享区来拿到我们的动态库。因此虽然我们指出了动态库路径,但是是给gcc指出的,而我们的操作系统并不知道动态库在哪,所以导致了运行的时候会报错。

那么我们该怎样正确的使用动态库呢?

方法一:将我们的动态库拷贝到系统路径中 

这个方法在静态库的制作那里演示过了,这里就不再演示了。

方法二:将动态库的路径添加到环境变量LD_LIBRARY_PATH中

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:动态库路径

注意:一定要加上之前的环境变量并且带上冒号进行分隔,否则之前的环境变量会被覆盖掉。 

所以我们的动态库路径就添加进环境变量LD_LIBRARY_PATH了。

那么我们再运行一下试试

运行成功了

但是这个方法有个缺点,它只在本次登录有效,你下次登录时你添加的环境变量就会被抹去。

方法三:创建.conf文件将动态库的路径添加到/etc/ld.so.conf.d/目录下

1、创建hjx.conf文件 

2、将动态库路径添加到/etc/ld.so.conf.d/hjx.conf中

3、ldconfig将我们的配置文件生效一下

这个方法可以永久有效

方法四:建立软链接

除了以上的方法外,还有其它方式可以用比如修改系统的配置文件等等。有兴趣的小伙伴可以自己去了解一下。 

最后再来回答最后一个问题:为什么要有库呢?

站在使用库的角度,库的存在可以减少我们的开发周期,极大地提高开发效率,避免重复造轮子,同时也能够利用已有的经过测试和优化的代码提高系统的性能和稳定性。

站在编写库的人的角度,库使用起来非常简单,方便开发人员进行快速开发,并且库中的实现对方看不到,从而实现了代码的安全。此外库只需提供一些标准的接口和协议,就可以使不同的组件和系统之间能够进行有效的通信和交互。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

实习算法准备之BFSDFS

这里写目录标题 1 理论1.1 BFS框架 2 例题2.1 二叉树的最小高度2.2 打开转盘锁2.3 滑动谜题 1 理论 BFS和DFS是两个遍历算法&#xff0c;其中DFS之前已经接触过&#xff0c;就是回溯&#xff0c;忘记的话请回顾回溯篇的例题&#xff08;全排列&#xff0c;N皇后&#xff09; B…

力扣数据库题库学习(4.23日)

610. 判断三角形 问题链接 解题思路 题目要求&#xff1a;对每三个线段报告它们是否可以形成一个三角形。以 任意顺序 返回结果表。 对于三个线段能否组成三角形的判定&#xff1a;任意两边之和大于第三边&#xff0c;对于这个表内的记录&#xff0c;要求就是&#xff08;x…

python学习笔记B-11:序列结构之列表--二维列表的遍历和生成式

二维列表的遍历方式&#xff0c;使用双层for循环&#xff0c;遍历索引号。 二维列表的生成式&#xff0c;也是使用类似双层循环的形式生成。 print("##初始化二维列表&#xff0c;每个元素就是1个列表") lst [["东方延续","太空军自然选择号舰长&qu…

【Java GUI】人机对弈五子棋

在学校的Java课程中&#xff0c;我们被分配了一项有趣的任务&#xff1a;开发一款能够实现人机对弈的五子棋游戏。为了更好地理解Java GUI的运用&#xff0c;并与大家分享学习心得&#xff0c;我将整个开发过程记录在这篇博客中。欢迎大家阅读并提供宝贵的意见和建议&#xff0…

PSoc™62开发板之SPI显示屏

实验目的 使用PSoc62™开发板驱动OLED模块&#xff0c;显示字符串、中文、数字 实验准备 PSoc62™开发板SSD1309 OLED模块 模块电路 引脚对应关系如下&#xff0c;这次实验采用的是SPI的驱动方式&#xff0c;可以无视SDA、SCL的命名 开发板GPIOSSD1309A0P10.0CSA1P10.1DC…

锂电池SOH预测 | 基于CNN的锂电池SOH预测(附matlab完整源码)

锂电池SOH预测 锂电池SOH预测完整代码锂电池SOH预测 锂电池的SOH(状态健康度)预测是一项重要的任务,它可以帮助确定电池的健康状况和剩余寿命,从而优化电池的使用和维护策略。 SOH预测可以通过多种方法实现,其中一些常用的方法包括: 容量衰减法:通过监测电池的容量衰减…

Python脚本抢票【笔记】

Python脚本抢票【笔记】 前言版权推荐Python脚本抢票【Python】microsoft edge驱动器下载以及使用最后 前言 2024-4-17 18:19:15 以下内容源自《【笔记】》 仅供学习交流使用 版权 禁止其他平台发布时删除以下此话 本文首次发布于CSDN平台 作者是CSDN日星月云 博客主页是ht…

Spring Web MVC入门(2)——请求

目录 一、传递单个参数 基础类型和包装类型的区别 1、基础类型 &#xff08;1&#xff09;不传参 &#xff08;2&#xff09;传字符串 2、包装类型 &#xff08;1&#xff09;不传参 &#xff08;2&#xff09;传字符串 3、小结 二、传递多个参数 三、传递对象 四、…

Fast-DetectGPT 无需训练的快速文本检测

本文提出了一种新的文本检测方法 ——Fast-DetectGPT&#xff0c;无需训练&#xff0c;直接使用开源小语言模型检测各种大语言模型&#xff0c;如GPT等生成的文本内容。 Fast-DetectGPT 将检测速度提高了 340 倍&#xff0c;将检测准确率相对提升了 75%&#xff0c;超过商用系…

Redis缓存问题:穿透,击穿,雪崩,双写一致性等

Redis缓存问题:穿透,击穿,雪崩,双写一致性等 在高并发场景下,数据库往往是最薄弱的环节,我们通常选择使用redis来进行缓存,以起到缓冲作用,来降低数据库的压力,但是一旦缓存出现问题,也会导致数据库瞬间压力过大甚至崩溃,从而导致整个系统崩溃.今天就聊聊常见的redis缓存问题.…

多路递归的一些算法题

前言 首先我想讲一下&#xff0c;我对多路递归的理解吧&#xff0c;我认为多路递归就是循环中套回调&#xff0c;对于循环有几次就是几叉树&#xff0c;就好比我们常用的二叉树的dfs(node.left) 和 dfs(node.right)等前中后序遍历&#xff0c;也就是for (int i 0; i < 2; …

AIGC - SD(中英文本生成图片) + PaddleHub/HuggingFace + stable-diffusion-webui

功能 stable-diffusion(文本生成图片)webui-win搭建&#xff08;开启api界面汉化&#xff09;PaddleHubHuggingFace: SD2&#xff0c;中文-alibaba/EasyNLP stable-diffusion-webui 下载与安装 环境相关下载 python&#xff08;文档推荐&#xff1a;Install Python 3.10.6 …

区块链技术与应用学习笔记(1-4节)——北大肖臻课程

目录 1. 区块链初识(课程简介&#xff09; 被过度炒作&#xff0c;落地应用有限&#xff1f; 下一代的价值互联网&#xff1f;世界上最慢的数据库&#xff1f; 2. BTC-密码学原理&#xff08;比特币&#xff09; 1)哈希 哈希函数特点 个人学习所得 2)签名 个人对于…

U-boot 21.10 启动流程梳理

目录 背景平台启动入口确认启动源码DuoS_SG2000_RISCVLubancat2_RK3568_ARM 初始化流程board_init_fboard_init_r 参考 背景 设备&#xff1a;MilkV Duo S 版本&#xff1a;U-boot 2021.10 编译命令 # Milkv-DuoS SD卡版本&#xff0c;对应[board]与[config]分别为&#xff1…

Leetcode_相交链表

✨✨所属专栏&#xff1a;LeetCode刷题专栏✨✨ ✨✨作者主页&#xff1a;嶔某✨✨ 题目&#xff1a; 题解&#xff1a; 看到这个题目首先我们要排除链表逆置的想法&#xff0c;如图、因为c1节点只有一个next指针&#xff0c;逆置后不可能同时指向a2和b3节点。 其次有的的同学…

24深圳杯AC题完整思路+可执行代码+参考论文!!!!

比赛题目的完整版思路可执行代码数据参考论文都会在第一时间更新上传的&#xff0c;大家可以参考我往期的资料&#xff0c;所有的资料数据以及到最后更新的参考论文都是一次付费后续免费的。注意&#xff1a;&#xff08;建议先下单占坑&#xff0c;因为随着后续我们更新资料数…

【AIGC调研系列】大型语言模型如何减少幻觉生成

在解读大型语言模型&#xff08;LLMs&#xff09;中的长格式事实性问题时&#xff0c;我们首先需要认识到这些模型在生成内容时可能会产生与既定事实不一致的情况&#xff0c;这种情况通常被称为“幻觉”[2][3]。这种现象不仅可能导致信息的误传&#xff0c;还可能对社会造成误…

新时代凌迟:考研

我不喜欢上班&#xff0c;但我很欣赏老板的品味&#xff0c;因为咱们公司竟然还在订阅报纸&#xff0c;而且只有一份&#xff0c;《中国青年报》。 这份报纸我最喜欢看的是“冰点周刊”专栏&#xff0c;因为这个栏目能让读者相信&#xff1a;报纸远远可以超越一天的生命。 昨天…

跨境代买淘宝系统,跨境代采系统,淘宝代购系统,淘宝代购集运系统,1688代采系统

淘宝代购系统是一种集成的电商平台服务&#xff0c;主要针对海外用户提供购买中国大陆商品的便利通道。以下是其核心功能与特点&#xff1a; 多语言支持&#xff1a;为了满足全球用户的需求&#xff0c;代购系统提供多语言界面&#xff0c;让不同国家和地区的用户都能方便地浏…

c++初阶——类和对象(下)

大家好&#xff0c;我是小锋&#xff0c;今天我们来学习我们类和对象的最后一个章节&#xff0c;我们本期的内容主要是类和对象的一些细节进行讲解 再谈构造函数 我们在初始化时有两种方式一种是函数体内初始化&#xff0c;一种是初始化列表 我们先来看看日期类的初始化 构造…