您现在的位置是:首页 >技术教程 >Linux:软硬链接和动静态库网站首页技术教程
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]== '
')
    {
        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 != '