第四章、文件和目录
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