第四章、文件和目录

4.2、函数stat、fstat、fstatat和lstat

stat 和 lstat 主要用于获取指定路径文件的元数据,其中 lstat 还能获取符号链接文件本身的信息。
fstat 和 fstatat 通过文件描述符获取文件的元数据,其中 fstatat 还允许在给定的目录中查找文件。


int stat(const char *pathname, struct stat *buf);
pathname: 文件路径名。
buf: 指向用于存储文件元数据的 stat 结构的指针。

int fstat(int fd, struct stat *buf);
fd: 文件描述符,指向要获取元数据的文件。
buf: 指向用于存储文件元数据的 stat 结构的指针。

int fstatat(int dirfd, const char *pathname, struct stat *buf, int flags);
dirfd: 目录文件描述符,指定要在其所在目录中查找文件的文件描述符。
pathname: 文件路径名。
buf: 指向用于存储文件元数据的 stat 结构的指针。
flags: 控制查找行为的标志,通常为 0。

int lstat(const char *pathname, struct stat *buf);
pathname: 文件路径名。
buf: 指向用于存储文件元数据的 stat 结构的指针。

struct stat {
    dev_t     st_dev;         // 文件所在设备的 ID
    ino_t     st_ino;         // 文件的 inode 号
    mode_t    st_mode;        // 文件的权限和类型
    nlink_t   st_nlink;       // 文件的硬链接数
    uid_t     st_uid;         // 文件的所有者用户 ID
    gid_t     st_gid;         // 文件的所有者组 ID
    dev_t     st_rdev;        // 如果文件是特殊文件(如设备文件),保存设备 ID
    off_t     st_size;        // 文件的大小(以字节为单位)
    blksize_t st_blksize;     // 文件系统 I/O 缓冲区大小
    blkcnt_t  st_blocks;      // 文件所占的块数
    struct timespec st_atim;  // 文件的最后访问时间
    struct timespec st_mtim;  // 文件的最后修改时间
    struct timespec st_ctim;  // 文件的最后状态改变时间
};

stat

root@VM-16-14-ubuntu:~/note# cat stat.c
#include "stdio.h"
#include "apue.h"

int
main()
{
    int rt;
    struct stat buf;

    rt = stat("/root/note/test", &buf);

    printf("%d\n", buf.st_uid);

    return 0;
}

root@VM-16-14-ubuntu:~/note# chown 9000:9000 test
root@VM-16-14-ubuntu:~/note# ./stat 
9000
root@VM-16-14-ubuntu:~/note# chown root test
root@VM-16-14-ubuntu:~/note# ./stat 
0

fstat

root@VM-16-14-ubuntu:~/note# cat fstat.c 
#include "stdio.h"
#include <fcntl.h>
#include "apue.h"

int
main()
{
    int fd, rt;
    struct stat buf;

    if ((fd = open("/root/note/test", 0)) == -1)
        printf("no file test error!");
    if ((rt = fstat(fd, &buf)) == -1)
        printf("fstat error!");

    printf("%d\n", buf.st_uid);

    return 0;
}

root@VM-16-14-ubuntu:~/note# ll test
-rwxrwxrwx 1 9000 9000 6 May 31 10:11 test*
root@VM-16-14-ubuntu:~/note# ll test_link 
lrwxrwxrwx 1 root root 4 May 31 11:15 test_link -> test*
root@VM-16-14-ubuntu:~/note# ./fstatat 
0
root@VM-16-14-ubuntu:~/note# vim fstatat.c 
root@VM-16-14-ubuntu:~/note# gcc fstatat.c -o fstatat
root@VM-16-14-ubuntu:~/note# ./fstatat 
9000

lstat

root@VM-16-14-ubuntu:~/note# cat lstat.c 
#include "stdio.h"
#include "apue.h"
#include "fcntl.h"

int
main()
{
    struct stat buf;

    if ((lstat("/root/note/test_link", &buf)) == -1)
        printf("lstat error!");

    printf("%d\n", buf.st_uid);
    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc lstat.c -o lstat
root@VM-16-14-ubuntu:~/note# ./lstat 
0
root@VM-16-14-ubuntu:~/note# ll test_link 
lrwxrwxrwx 1 root root 4 May 31 11:15 test_link -> test*


4.7、函数access和faccessat

access 和 faccessat 是用于检查文件的访问权限的系统调用函数。它们在 UNIX 和类 UNIX 操作系统中被广泛使用。


int access(const char *pathname, int mode);
pathname: 文件路径名。
mode: 要检查的权限模式,可以是以下常量的组合:

    R_OK(可读)
    W_OK(可写)
    X_OK(可执行)
    F_OK(存在)

int faccessat(int dirfd, const char *pathname, int mode, int flags);
dirfd: 目录文件描述符,指向要在其所在目录中查找文件的文件描述符。
    如果 dirfd 是 AT_FDCWD,则相对于当前工作目录进行操作。
pathname: 文件路径名。
mode: 要检查的权限模式,含义与 access 的 mode 参数相同。
flags: 控制检查行为的标志,可以是以下值的组合:
    AT_EACCESS:检查有效用户 ID 和组 ID,而不是实际的用户 ID 和组 ID。
    AT_SYMLINK_NOFOLLOW:如果 pathname 是符号链接,则不跟随它。

access

root@VM-16-14-ubuntu:~/note# cat access.c 
#include "apue.h"
#include "fcntl.h"
#include "stdio.h"

int
main(int argc, char *argv[]) //argc命令行参数的数量, argv命令行每个参数的内容,argv[0]为命令本身
{
    if (argc != 2)
        printf("usage: a.out <pathname>");
    if (access(argv[1], R_OK) < 0) //是否可读
        printf("access error for %s", argv[1]);
    else
        printf("read access OK\n");
    if (open(argv[1], O_RDONLY) < 0) //O_RDONLY: 以只读模式打开文件。
        printf("open error for %s", argv[1]);
    else
        printf("open for reading OK\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# ./access test
read access OK
open for reading OK

faccessat

root@VM-16-14-ubuntu:~/note# cat faccessat.c 
#include "apue.h"
#include "fcntl.h"
#include "stdio.h"

int
main()
{
    int fd;
    if ((fd = open("/root/note/", 0)) == -1)
        printf("open error!");
    if (faccessat(fd, "test_link", R_OK, AT_SYMLINK_NOFOLLOW) == -1)
        printf("faccessat error!");
    else
        printf("faccessat access ok\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc faccessat.c -o faccessat
root@VM-16-14-ubuntu:~/note# ./faccessat 
faccessat access ok

4.8、函数umask

屏蔽字也称反掩码

mode_t umask(mode_t mask);

mask: 新的 umask 值。如果调用 umask 时传递了 mask 参数,函数将返回之前的 umask 值并设置新的 umask 值为 mask。如果只想获取当前 umask 值,可以传递 0,然后用返回值设置新的 umask 值。

root@VM-16-14-ubuntu:~/note# cat umask.c 
#include "apue.h"
#include "fcntl.h"
#include "stdio.h"

#define RWRWRW (S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)

int
main()
{
    umask(0);
    if (creat("foooo", RWRWRW) < 0)
        printf("creat error for foooo");
    umask(S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
    if (creat("bar", RWRWRW) < 0)
        printf("creat error for bar");
    return 0;   
}

root@VM-16-14-ubuntu:~/note# ./umask 
root@VM-16-14-ubuntu:~/note# ll foooo 
-rw-rw-rw- 1 root root 0 May 31 16:33 foooo
root@VM-16-14-ubuntu:~/note# ll bar 
-rw------- 1 root root 0 May 31 16:33 bar

4.9、函数chmod、fchmod和fchmodat

chmod、fchmod 和 fchmodat 是用于修改文件或目录权限的系统调用函数和命令。在 UNIX 和类 UNIX 操作系统中,这些函数和命令可以用来设置文件或目录的读、写、执行权限。

int chmod(const char *pathname, mode_t mode);
pathname: 文件路径名。
mode: 新的权限设置,使用 mode_t 类型。

int fchmod(int fd, mode_t mode);
fd: 文件描述符。
mode: 新的权限设置,使用 mode_t 类型

int fchmodat(int dirfd, const char *pathname, mode_t mode, int flags);
dirfd: 目录文件描述符。如果为 AT_FDCWD,则相对于当前工作目录。
pathname: 文件路径名。
mode: 新的权限设置,使用 mode_t 类型。
flags: 标志,通常为 0。可以使用 AT_SYMLINK_NOFOLLOW 指示不跟随符号链接。

chmod

root@VM-16-14-ubuntu:~/note# cat chmod.c 
#include "apue.h"
#include "stdio.h"

int
main()
{
    mode_t mode = S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH;

    if ((chmod("/root/note/test", mode)) == -1)
        printf("chmod error");

    return 0;
}

root@VM-16-14-ubuntu:~/note# chmod 000 test
root@VM-16-14-ubuntu:~/note# ll test
---------- 1 9000 9000 6 May 31 10:11 test
root@VM-16-14-ubuntu:~/note# ./chmod 
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 9000 9000 6 May 31 10:11 test

fchmod

root@VM-16-14-ubuntu:~/note# cat fchmod.c 
#include "stdio.h"
#include "fcntl.h"
#include "apue.h"

int
main(int argc, char** argv)
{
    int fd;
    mode_t mode[256];
    sscanf(argv[2], "%o", mode);

    if (argc < 2)
        printf("no file\n");
    if ((fd = open(argv[1], 0)) == -1)
        printf("open error!\n");
    if ((fchmod(fd, *mode)) == -1)
        printf("fchmod error!\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc fchmod.c -o fchmod
root@VM-16-14-ubuntu:~/note# ./fchmod test 0644
root@VM-16-14-ubuntu:~/note# ll test
-rw-r--r-- 1 9000 9000 6 May 31 10:11 test

fchmodat

root@VM-16-14-ubuntu:~/note# cat fchmodat.c 
#include "stdio.h"
#include "fcntl.h"
#include "apue.h"

int
main()
{
    int fd;
    mode_t mode = S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH;

    if((fd = open("/root/note", O_EXCL)) == -1)
        printf("open file error!");
    if (fchmodat(fd, "test", mode, 0) == -1)
        printf("fchmodat error!");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc fchmodat.c -o fchmodat
root@VM-16-14-ubuntu:~/note# chmod 000 test
root@VM-16-14-ubuntu:~/note# ./fchmodat 
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 9000 9000 6 May 31 10:11 test

4.11、函数chown、fchown、fchownat和lchown

chown:用于命令行或程序中直接更改文件或目录的所有者和组。
fchown:通过文件描述符更改文件的所有者和组。
fchownat:允许在指定目录中查找文件并更改其所有者和组。
lchown:更改符号链接本身的所有者和组。

int chown(const char *pathname, uid_t owner, gid_t group);
pathname: 文件路径名。
owner: 新的所有者的用户 ID。
group: 新的组的组 ID。

int fchown(int fd, uid_t owner, gid_t group);
fd: 文件描述符。
owner: 新的所有者的用户 ID。
group: 新的组的组 ID。

int fchownat(int dirfd, const char *pathname, uid_t owner, gid_t group, int flags);
dirfd: 目录文件描述符。如果为 AT_FDCWD,则相对于当前工作目录。
pathname: 文件路径名。
owner: 新的所有者的用户 ID。
group: 新的组的组 ID。
flags: 控制行为的标志,可以是 0 或 AT_SYMLINK_NOFOLLOW,后者表示不跟随符号链接。

int lchown(const char *pathname, uid_t owner, gid_t group);
pathname: 符号链接路径名。
owner: 新的所有者的用户 ID。
group: 新的组的组 ID。
root@VM-16-14-ubuntu:~/note# cat chown.c 
#include "apue.h"
#include "stdio.h"
#include "fcntl.h"

#define MAX 256

int
main(int argc, char *argv[])
{
    uid_t uid[MAX];
    gid_t gid[MAX];

    sscanf(argv[1], "%u", uid);
    sscanf(argv[2], "%u", gid);

    chown(argv[3], *uid, *gid);

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc chown.c -o chown
root@VM-16-14-ubuntu:~/note# ./chown 8000 8000 test
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 8000 8000 6 May 31 10:11 test
root@VM-16-14-ubuntu:~/note# cat fchown.c
#include "stdio.h"
#include "fcntl.h"
#include "apue.h"

int
main()
{
    int fd;

    if ((fd = open("/root/note/test", 0)) == -1)
        printf("open error!");
    if (fchown(fd, 9000, 9000) == -1)
        printf("fchown error!");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc fchown.c -o fchown
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 8000 8000 6 May 31 10:11 test
root@VM-16-14-ubuntu:~/note# ./fchown 
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 9000 9000 6 May 31 10:11 test

root@VM-16-14-ubuntu:~/note# cat fchownat.c 
#include "stdio.h"
#include "apue.h"
#include "fcntl.h"

int
main()
{
    int fd;
    if ((fd = open("/root/note", 0)) == -1)
        printf("open file error!\n");

    if (fchownat(fd, "test", 8888, 8888, 0) == -1)
        printf("fchownat error!\n");

    return 0;   
}

root@VM-16-14-ubuntu:~/note# gcc fchownat.c -o fchownat
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 9000 9000 6 May 31 10:11 test
root@VM-16-14-ubuntu:~/note# ./fchownat 
root@VM-16-14-ubuntu:~/note# ll test
-rw-rw-rw- 1 8888 8888 6 May 31 10:11 test
root@VM-16-14-ubuntu:~/note# cat lchown.c 
#include "stdio.h"
#include "apue.h"
#include "fcntl.h"

int
main()
{
    if (lchown("/root/note/test_link", 9999, 9999) == -1)
        printf("lchown test_link error!\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc lchown.c -o lchown
root@VM-16-14-ubuntu:~/note# ll test_link 
lrwxrwxrwx 1 root root 4 May 31 11:15 test_link -> test
root@VM-16-14-ubuntu:~/note# ./lchown 
root@VM-16-14-ubuntu:~/note# ll test_link 
lrwxrwxrwx 1 9999 9999 4 May 31 11:15 test_link -> test

4.15、函数link、linkat、unlink、unlinkat和remove

link 和 linkat 用于创建硬链接。
unlink 和 unlinkat 用于解除连接文件。
remove 是一个标准库函数,可以解除连接文件或空目录。

int link(const char *oldpath, const char *newpath);
oldpath:现有文件的路径。
newpath:新硬链接的路径。

int linkat(int olddirfd, const char *oldpath, int newdirfd, const char *newpath, int flags);
olddirfd:指向 oldpath 的目录文件描述符。
oldpath:现有文件的路径。
newdirfd:指向 newpath 的目录文件描述符。
newpath:新硬链接的路径。
flags:标志,通常为 0

int unlink(const char *pathname);
pathname:要解除连接的文件路径。

int unlinkat(int dirfd, const char *pathname, int flags);
dirfd:指向 pathname 的目录文件描述符。
pathname:要解除连接的文件路径。
flags:标志,可以是 0 或 AT_REMOVEDIR(删除目录)

int remove(const char *pathname);
pathname:要解除连接的文件或目录路径。

link

root@VM-16-14-ubuntu:~/note# cat link.c 
#include "apue.h"
#include <fcntl.h>
#include <stdio.h>

int 
main()
{
    int rt;
    if ((rt = link("fooo", "fooo_link")) != 0)
        printf("no fooo file or create fooo_link error");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc link.c -o link
root@VM-16-14-ubuntu:~/note# touch fooo
root@VM-16-14-ubuntu:~/note# ./link 
root@VM-16-14-ubuntu:~/note# stat fooo_link 
  File: fooo_link
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 2
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 15:15:26.320859577 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800

linkat

root@VM-16-14-ubuntu:~/note# cat linkat.c 
#include <stdio.h>
#include "apue.h"
#include "fcntl.h"

int
main()
{
    int fd_old, fd_new;
    if ((fd_old = open("/root/note", 0)) < 0)
        printf("no path /root/note %d\n", fd_old);
    if ((fd_new = open("/root/test", 0)) < 0)
        printf("no path /root/test %d\n", fd_new);
    if (linkat(fd_old, "fooo", fd_new, "fooo_link", 0) != 0)
        printf("create link file fooo_link error\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# ./linkat 
root@VM-16-14-ubuntu:~/note# stat ../test/fooo_link 
  File: ../test/fooo_link
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 3
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 15:31:18.328331439 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800

unlink


root@VM-16-14-ubuntu:~/note# cat unlink.c
#include <stdio.h>
#include "apue.h"

int
main()
{
    if (unlink("fooo_link") != 0)
        printf("no file link fooo_link error");

    return 0;
}

root@VM-16-14-ubuntu:~/note# stat fooo_link 
  File: fooo_link
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 3
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 15:31:18.328331439 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800
root@VM-16-14-ubuntu:~/note# ./unlink 
root@VM-16-14-ubuntu:~/note# stat fooo_link 
stat: cannot statx 'fooo_link': No such file or directory

unlinkat

root@VM-16-14-ubuntu:~/note# cat unlinkat.c 
#include "apue.h"
#include <stdio.h>
#include <fcntl.h>

int
main()
{
    int fd;
    if ((fd = open("/root/test", 0)) < 0)
        printf("error open path /root/test\n");
    if (unlinkat(fd, "fooo_link", 0) != 0)
        printf("error unlink fooo_link file\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc unlinkat.c -o unlinkat
root@VM-16-14-ubuntu:~/note# stat ../test/fooo_link 
  File: ../test/fooo_link
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 2
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 17:20:32.649056202 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800
root@VM-16-14-ubuntu:~/note# ./unlinkat 
root@VM-16-14-ubuntu:~/note# stat ../test/fooo_link 
stat: cannot statx '../test/fooo_link': No such file or directory

remove

root@VM-16-14-ubuntu:~/note# cat remove.c
#include <stdio.h>
#include "fcntl.h"
#include "apue.h"

int
main()
{
    if (remove("fooo_link") != 0)
        printf("error remove fooo_link file\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc remove.c -o remove
root@VM-16-14-ubuntu:~/note# ln fooo fooo_link
root@VM-16-14-ubuntu:~/note# ./remove 
root@VM-16-14-ubuntu:~/note# stat fooo_link
stat: cannot statx 'fooo_link': No such file or directory

4.16、函数rename和renameat

int rename(const char *oldpath, const char *newpath);
oldpath: 旧文件路径。
newpath: 新文件路径。

int renameat(int olddirfd, const char *oldpath, int newdirfd, const char *newpath);
olddirfd: oldpath 所在目录的文件描述符。如果传递 AT_FDCWD,则 oldpath 相对于当前工作目录。
oldpath: 旧文件路径。
newdirfd: newpath 所在目录的文件描述符。如果传递 AT_FDCWD,则 newpath 相对于当前工作目录。
newpath: 新文件路径。

rename

root@VM-16-14-ubuntu:~/note# cat rename.c 
#include "apue.h"
#include <stdio.h>
#include "fcntl.h"

int
main()
{
    if (rename("fooo", "fooo_bak") != 0)
        printf("error rename fooo");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc rename.c -o rename
root@VM-16-14-ubuntu:~/note# stat fooo
  File: fooo
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 1
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 17:31:36.536588164 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800
root@VM-16-14-ubuntu:~/note# ./rename 
root@VM-16-14-ubuntu:~/note# stat fooo
stat: cannot statx 'fooo': No such file or directory
root@VM-16-14-ubuntu:~/note# stat fooo_bak 
  File: fooo_bak
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 1
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 17:41:24.364187653 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800

renameat

root@VM-16-14-ubuntu:~/note# cat renameat.c 
#include <stdio.h>
#include "fcntl.h"
#include "apue.h"

int
main()
{
    int oldfd, newfd;
    if ((oldfd = open("/root/note", 0)) < 0 || (newfd = open("/root/test", 0)) < 0)
        printf("error no /root/note or /root/test\n");

    if (renameat(oldfd, "fooo_bak", newfd, "fooo") != 0)
        printf("error rename\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc renameat.c -o renameat
root@VM-16-14-ubuntu:~/note# stat fooo_bak 
  File: fooo_bak
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 1
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 17:41:24.364187653 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800

root@VM-16-14-ubuntu:~/note# stat ../test/fooo
stat: cannot statx '../test/fooo': No such file or directory
root@VM-16-14-ubuntu:~/note# ./renameat 
root@VM-16-14-ubuntu:~/note# stat ../test/fooo
  File: ../test/fooo
  Size: 0           Blocks: 0          IO Block: 4096   regular empty file
Device: fc02h/64514d    Inode: 792643      Links: 1
Access: (0644/-rw-r--r--)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 15:15:21.684862154 +0800
Modify: 2024-06-11 15:15:21.684862154 +0800
Change: 2024-06-11 18:18:57.710558312 +0800
 Birth: 2024-06-11 15:15:21.684862154 +0800

4.18、函数symlink和symlinkat

int symlink(const char *target, const char *linkpath);
target: 目标文件或目录的路径。
linkpath: 符号链接文件的路径。

int symlinkat(const char *target, int newdirfd, const char *linkpath);
target: 目标文件或目录的路径。
newdirfd: 符号链接所在目录的文件描述符。如果传递 AT_FDCWD,则 linkpath 相对于当前工作目录。
linkpath: 符号链接文件的路径,相对于 newdirfd 指定的目录。

symlink

root@VM-16-14-ubuntu:~/note# cat symlink.c 
#include "apue.h"
#include <stdio.h>

int
main()
{
    if (symlink("fooo", "fooo_slk") != 0)
        printf("error symlink fooo");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc symlink.c -o symlink
root@VM-16-14-ubuntu:~/note# touch fooo
root@VM-16-14-ubuntu:~/note# ./symlink 
root@VM-16-14-ubuntu:~/note# stat fooo_slk 
  File: fooo_slk -> fooo
  Size: 4           Blocks: 0          IO Block: 4096   symbolic link
Device: fc02h/64514d    Inode: 792662      Links: 1
Access: (0777/lrwxrwxrwx)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 18:27:35.926209089 +0800
Modify: 2024-06-11 18:27:31.746211668 +0800
Change: 2024-06-11 18:27:31.746211668 +0800
 Birth: 2024-06-11 18:27:31.746211668 +080

symlinkat

root@VM-16-14-ubuntu:~/note# cat symlinkat.c 
#include <stdio.h>
#include "apue.h"
#include "fcntl.h"

int
main()
{
    int fd;
    if ((fd = open("/root/test", 0)) < 0)
        printf("error open /root/test\n");

    if (symlinkat("fooo", fd, "fooo_slkt") != 0)
        printf("error symlinkat fooo_slkt\n");

    return 0;
}

root@VM-16-14-ubuntu:~/note# gcc symlinkat.c -o symlinkat
root@VM-16-14-ubuntu:~/note# ./symlinkat 
root@VM-16-14-ubuntu:~/note# stat ../test/fooo_slkt 
  File: ../test/fooo_slkt -> fooo
  Size: 4           Blocks: 0          IO Block: 4096   symbolic link
Device: fc02h/64514d    Inode: 655532      Links: 1
Access: (0777/lrwxrwxrwx)  Uid: (    0/    root)   Gid: (    0/    root)
Access: 2024-06-11 18:32:08.942040933 +0800
Modify: 2024-06-11 18:32:01.430045552 +0800
Change: 2024-06-11 18:32:01.430045552 +0800
 Birth: 2024-06-11 18:32:01.430045552 +0800

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注