• 第4章 文件和目录(3)_文件系统结构及软、硬链接


    3. Linux文件系统结构

    3.1 文件系统结构

     

     (1)超级块

      文件系统中的第一个块被称为超级块。这个块存放文件系统本身的信息。比如,记录了每个区域的大小,也存放了未被使用的磁盘块的信息。其包含的信息如下:

    主要域

    含义

    Magic Number

    用来检验是否是一个真正的EXT2文件系统超级块

    Revision Level

    主从修订版本号。让安装代码据此判断文件系统是否支持只存在于某个特定版本文件系统中的属性。

    Block Group Number

    超级块的组数量

    Block Size

    以字节记数的文件系统块大小,如1024字节

    Blocks per Group

    每个组中块的数目。当文件系统创建时,这个大小就被固定下来。

    Free Blocks

    文件系统中空闲块数

    Free Inodes

    文件系统中空闲的inode数

    First Inode

    文件系统中第一个inode号。一般就是根分区“/”的目录入口

    (2)inode表:

      每个子目录和文件只有唯一的一个 inode 块。它包含了文件系统中文件的基本属性(文件的长度、创建及修改时间、权限、所属关系、硬链接数指向数据块位置的指针链表等相关信息

    (3) 数据区

      文件系统的第三个部分是数据区。文件的内容保存在这个区域。若文件太大时,会占用多个 block。为了提高目录访问效率,Linux 还提供了表示路径与 inode 对应关系的目录块(dentry结构)。它描述了路径信息并连接到节点 inode,它包括各种目录信息,还指向了 inode 和超级块。大小为 4096字节,而数据块记录着具体内容。注意Linux中文件和目录本质都是文件,所以它们本身的数据都存储在数据区中。

    3.2 在文件系统中搜寻文件:以搜索/var/test.txt为例

     

    (1)从超级块中拿到根目录inode节点信息(inode为2)。

    (2)然后根据根目录的inode找到根目录项,从中找到var的inode节点。

    (3)根据var的inode(786433)找到var的目录项,从中找到test.txt文件的inode节点(796326)。

    (4)最后根据test.txt的inode节点信息,就可以找到test.txt文件所在的数据块。

    3.3 硬链接和软链接

    (1)硬链接的操作函数:link和ulink函数

    头文件

    #include<unistd.h>

    函数

    int link(const char* existingpath, const char* newpath); //创建硬链接

    int unlink(const char* pathname); //删除链接(软、硬链接均可删除)

    返回值

    若成功则为0,若出错则为-1

    功能

    创建一个指向现有文件的硬链接或删除指定的硬链接。

    备注

    (1)硬链接创建的条件

      ①针对文件创建链接;②必须是同一个分区(不能跨文件系统);③只有超级用户才能对目录建立链接。

    (2)文件删除条件

      ①链接计数为0;②无其他进程打开文件

    【编程实验】创建硬链接

    //file_hardlink.c

    #include <unistd.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <fcntl.h>
    
    //为文件创建多个硬链接
    
    int main(int argc, char* argv[])
    {
        if(argc < 2){
            fprintf(stderr, "usage: %s srcfile files
    ", argv[0]);
            exit(1);
        }
    
        int i=0;
        for(i=2; i<argc; i++){
            if(link(argv[1], argv[i]) < 0 ){
                perror("link error");
                continue;
            }
        }
    
        printf("create success
    ");
    
        return 0;
    }

    (2)软链接的操作函数:symlink和readlink函数

    symlink函数

    头文件

    #include<unistd.h>

    函数

    int symlink(const char* actualpath, const char* sympath);

    返回值

    若成功则为0,若出错则为-1

    功能

    创建一个符号链接(软链接)

    备注

    (1)创建符号链接并不要求actualpath存在。

    (2)可以跨文件系统建立符号链接

    radlink函数

    函数

    int readlink(const char* pathname, char* buf, size_t bufsize);

    返回值

    成功返回读到的字节数,出错返回-1

    功能

    打开该链接本身,并读该链接中的名字。

    备注

    open函数打开的是软链接引用的文件,而readlink打开的是软链接本身

    【编程实验】创建软链接

    //file_softlink.c

    #include <unistd.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <fcntl.h>
    #include <memory.h>
    
    int main(int argc, char* argv[])
    {
        if(argc < 3){
            fprintf(stderr, "usage: %s srcfile linkfile
    ", argv[0]);
            exit(1);
        }
      
        //对源文件创建符号链接(软链接)
        if(symlink(argv[1], argv[2]) < 0){
            perror("open error");
            exit(1);
        }
    
        char buffer[1024];
        ssize_t size;
        int fd = open(argv[2], O_RDONLY);
        if(fd < 0){
            perror("open error");
            exit(1);
        }
    
        //读取软链接文件引用的源文件内容
        memset(buffer, 0, sizeof(buffer));
        if((size = read(fd, buffer, sizeof(buffer))) < 0){
            perror("read error");
            exit(1);
        }else{
            write(STDOUT_FILENO, buffer, size);
        }
    
        close(fd);
    
        printf("
    ");
    
        //读取软链接本身的内容
        memset(buffer, 0, sizeof(buffer));
        if((size = readlink(argv[2], buffer, sizeof(buffer))) < 0 )
        {
            perror("readlink error");
            exit(1);
        }else{
            write(STDOUT_FILENO, buffer,size);
        }
    
        printf("
    ");
    
        return 0;
    }

    (3)软硬链接的区别与联系

     

      ①软链接有自己的文件属性及权限等,而硬链接只有目录项,没有独立的inode节点。

      ②可对不存在的文件或目录创建软链接。只不过,此时其数据块中的值不存放任何文件的路径而己。

      ③软链接可以交叉文件系统。但硬链接不可以跨文件系统,因为他们指向的是同一个inode节点。

      ④创建软链接时,源文件的计数不会增加。但为源文件创建硬链接时,计算每次加1。

      ⑤删除软链接并不影响指向的文件。但若被指向的原文件被删除(相当于上图中的test.txt被删除),则该软链接会因找不到这个test.txt而成为死链接。如果其所向的文件重新被恢复,则死链接可恢复为正常的软件链接。

      ⑥删除源文件或硬链接会使硬链接计数减1,当计数为0时,相应的数据块将被物理删除

    (4)remove和rename函数

    remove函数

    头文件

    #include<unistd.h>

    函数

    int remove(const char* pathname);

    返回值

    若成功则为0,若出错则为-1

    功能

    解除对一个文件或目录的链接

    备注

    (1)对于文件:remove的功能相当于unlink

    (2)对于目录:remove的功能相当于rmdir

    rename函数

    函数

    int renamelink(const char* oldname, const char* newname);

    返回值

    成功返回读到的字节数,出错返回-1

    功能

    文件或目录的更名

    备注

    open函数打开的是软链接引用的文件,而readlink打开的是软链接本身。

    3.4 文件时间

    (1)struct stat结构体中表示文件的3种时间

    字段

    说明

    例子

    ls选项

    st_atime

    文件数据最后访问时间

    read

    -u

    st_mtime

    文件数据最后修改时间

    write

    默认

    st_ctime

    i节点最后更改时间

    chmod,chown

    -c

    (2)utime函数

    头文件

    #include<sys/types.h>

    #include<utime.h>

    函数

    int utime(const char* pathname, struct utimbuf* times);

    返回值

    若成功则为0,若出错则为-1

    功能

    更改文件的存取和修改时间

    参数

    (1)utimbuf结构体

    struct utimbuf{
        time_t actime;     //access time
        time_t modtime     //modification time
    }

    (2)参数times

      ①NULL:由取当前时间(但要求进程的有效ID必须等于该文件的所有者ID;或者进程对该文件具有写权限)

      ②非空:取times结构体中的时间(要求进程的有效ID必须等于该文件的所有者ID;或者进程是超级用户进程)

    备注

    utime操作会自动更新st_ctime的值。

    【编程实验】更改文件时间

    //file_utime.c

    #include <time.h>
    #include <utime.h>
    #include <sys/stat.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <fcntl.h>
    #include <memory.h>
    
    //输出文件时间信息
    void out(struct stat* buff)
    {
        //ctime函数将时间转为字符串
        printf("atime: %s", ctime(&buff->st_atime));
        printf("mtime: %s", ctime(&buff->st_mtime));
        printf("ctime: %s", ctime(&buff->st_ctime));
    }
    
    //获得文件属性
    struct stat get_stat(char* file)
    {
        struct stat ret;
        memset(&ret, 0, sizeof(ret));
        if(lstat(file, &ret) < 0){
            perror("lstat error");
            exit;
        }
    
        return ret;
    }
    
    int main(int argc, char* argv[])
    {
        if (argc < 2){
            fprintf(stderr, "usage: %s file
    ", argv[0]);
            exit(1);
        }
    
        struct stat buff;
        buff = get_stat(argv[1]);
        
        //备份原先的时间
        struct stat backup = buff;
        //输出文件时间
        out(&buff);
        printf("-------------------------------
    ");
    
        //设置文件的时间为当前系统时间
        utime(argv[1], NULL);
        buff = get_stat(argv[1]);
        out(&buff);
        printf("-------------------------------
    ");
        
        //将文件时间恢复为原先时间
        struct utimbuf timebuf;
        timebuf.actime = backup.st_atime;
        timebuf.modtime = backup.st_mtime;
        utime(argv[1], &timebuf);
        buff = get_stat(argv[1]);
        out(&buff);
    
        return 0;
    }
    /*输出结果:
    atime: Sun Jan 22 19:04:24 2017
    mtime: Sun Jan 22 19:04:00 2017
    ctime: Sun Jan 22 19:58:39 2017   //注意,这个时间utime会自动更新
    -------------------------------
    atime: Sun Jan 22 20:01:08 2017
    mtime: Sun Jan 22 20:01:08 2017
    ctime: Sun Jan 22 20:01:08 2017   //注意,这个时间utime会自动更新
    -------------------------------
    atime: Sun Jan 22 19:04:24 2017
    mtime: Sun Jan 22 19:04:00 2017
    ctime: Sun Jan 22 20:01:08 2017   //注意,这个时间utime会自动更新
    */
  • 相关阅读:
    MapInfo 文件格式说明
    一个经典编程面试题的“隐退”
    Polar 投影c#版本移植
    关于 tf.nn.softmax_cross_entropy_with_logits 及 tf.clip_by_value
    系列解读Dropout
    python删除所有的中文字符、非ASCII或非英文字符,检查字符串是否包含非ASCII
    Python使用split使用多个字符分隔字符串
    Convolutional Neural Networks for Visual Recognition
    【TensorFlow】tf.nn.conv2d是怎样实现卷积的?
    [TensorFlow] tf.nn.softmax_cross_entropy_with_logits的用法
  • 原文地址:https://www.cnblogs.com/5iedu/p/6350318.html
Copyright © 2020-2023  润新知