link
unlink
书中:只有当链接计数达到0时,该文件的内容才可被删除。另一个条件也会阻止删除文件的内容–只要有进程打开了该文件,其内容也不能删除。关闭一个文件时,内核首先检查打开该文件的进程个数;如果这个计数达到 0,内核再去检查其链接计数;如果计数也是0,那么就删除该文件的内容。
mac实践不是如此。
尽管在进程中已经打开了该文件,在程序还没结束时,当unlink之后文件立马被删除了。
#include <apue.h>
#include <fcntl.h>
#define RWRWRW (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)
int main(int argc, char *argv[])
{
/* // 1. 测试文件类型
struct stat buf;
char *ptr;
for (int i = 1; i < argc; ++i)
{
printf("%s: ", argv[i]);
if (lstat(argv[i], &buf) < 0)
{
err_ret("lstat error");
continue;
}
if (S_ISREG(buf.st_mode))
ptr = "regular";
else if (S_ISDIR(buf.st_mode))
ptr = "directory";
else if (S_ISCHR(buf.st_mode))
ptr = "character special";
else if (S_ISBLK(buf.st_mode))
ptr = "block special";
else if (S_ISSOCK(buf.st_mode))
ptr = "socket";
else if (S_ISFIFO(buf.st_mode))
ptr = "FIFO";
else if (S_ISLNK(buf.st_mode))
ptr = "symbolic link";
else
ptr = "**unknown mode**";
printf("%s\n",ptr);
}*/
/* // 2. 测试文件的权限
if (argc != 2)
err_quit("usage: ./a.out <pathname>");
// 测试实际用户id 对该文件的读取权限
if (access(argv[1], R_OK) < 0)
err_ret("access error for %s", argv[1]);
else
printf("read access ok.\n");
// 测试进程的有效用户id 对该文件的读取权限
// 一般情况下有效用户id为实际用户id,但是可以通过设置用户id改变有效用户id为文件的所有者
if (open(argv[1], O_RDONLY) < 0)
err_ret("access error for %s", argv[1]);
else
printf("open for reading access ok.\n"); */
/* // 3. umask设置文件模式的创建屏蔽字 八进制表示3类用户的读写可执行权限,mac默认为022 rwxr-xr-x
umask(0); // 不屏蔽任何权限
if (creat("foo", RWRWRW) < 0)
err_sys("create error for foo");
umask(S_IROTH | S_IWOTH); // 屏蔽了其他用户的读写权限
if (creat("bar", RWRWRW) < 0)
err_sys("create error for bar");
// -rw-rw---- 1 wmj staff 0 1 2 20:17 bar
// -rw-rw-rw- 1 wmj staff 0 1 2 20:17 foo */
// 4. 当文件的链接数为0 且没有进程打开该文件的时候文件才会被关闭
if (open("foo1", O_RDWR) < 0)
err_sys("open error.");
if (unlink("foo1") < 0)
err_sys("unlink error.");
printf("file unlink\n");
sleep(20);
printf("done.\n");
exit(0);
}