APUE第四章文件和目录
与进程相关联的ID
1.实际用户ID+实际组ID
2.有效用户ID+有效组ID+附属组ID
3.保存的设置用户ID+保存的设置组ID
- 执行一个程序文件时,进程的有效用户ID=实际用户ID,有效组ID=实际组ID,
当我执行
./a.out
文件时,如果a.out
此文件权限是root root
,但是我以非root
用户来执行的(实际用户是非root
用户,此时进程的有效用户id变成实际用户id,那么该进程的权限只有非root
权限. - 但是可以修改文件的有效用户ID为该文件所有者的用户id(组同理),也就以非root用户身份执行文件时,文件拥有root权限 (文件本身属于root,通过修改
st_mode
中的特殊标志,设置用户ID位和设置组ID位) - 实例:
程序
passwd
允许任一用户改变口令,因为改程序能将新口令写入口令文件,而只有root才有对该文件的写权限,
函数
access
,按实际用户ID和实际组ID进行访问权限测试
1 |
|
umask
:为进程设置文件模式创建掩码,意思就是创建文件的权限位是在umask
码的前提下才有
1 |
|
- 相当于清空原本的
umask
值,重新设定,然后提权限
3.link
创建软连接和硬链接
1 |
|
递归降序遍历目录层次结构,按文件类型计数
1 |
|
课后练习
4.1 stat
跟随符号链接,lstat
不跟随符号链接
4.2 umask 777
则所有位被关闭,文件任何权限都没有
4.6
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
// 首先创建一个空洞文件
#include <bits/stdc++.h>
#include <unistd.h>
#include <fcntl.h>
#include "apue.h"
int main(int argc,char *argv[]) {
std::string path = "/home/luchao/Documents/Apue/4/1.in";
umask(0);
int fd = open(path.c_str(),O_CREAT|O_RDWR,0644);
if (fd == -1)
err_sys("open errro");
std::string content("this is write content");
if (write(fd,content.c_str(),content.size()) != content.size())
err_sys("write error");
// 重新定位形成空洞
int offset = lseek(fd,1024,SEEK_SET);
if (offset != 1024)
err_sys("offset error");
content = "other content";
if (write(fd,content.c_str(),content.size()) != content.size())
err_sys("write second error");
close(fd);
return 0;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
// 复制空洞文件
#include <bits/stdc++.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/stat.h>
const int maxn = 1024;
char buf[maxn];
int main(int argc,char *argv[]) {
if (argc != 3) {
std::cerr << "Usage: ./a.out file1 file2\n";
exit(1);
}
int fd1,fd2;
if ((fd1 = open(argv[1],O_RDWR)) == -1) {
std::cerr << "open " << argv[1] << " failed\n";
exit(1);
}
// 设置文件掩码
umask(0);
if ((fd2 = open(argv[2],O_RDWR|O_CREAT,0644)) == -1) {
std::cerr << "open " << argv[1] << " failed\n";
exit(1);
}
long long out_offset = 0;
int cnt,start_offset;
// =0是正是内容,!=0 是空洞
bool flag;
while (cnt = read(fd1,buf,maxn)) {
flag = 0,start_offset = 0;
for (int i=0; i<cnt; ++i) {
if (!flag && buf[i]!=0) {
flag = 1,start_offset = i;
if (lseek(fd2,out_offset + start_offset,SEEK_SET) == -1) {
std::cerr << "lseek failed\n";
exit(1);
}
}
// 空洞出现
else if (flag && buf[i] == 0) {
// 写入
if (write(fd2,buf + start_offset,i - start_offset) == -1) {
std::cerr << "fd2 write failed\n";
exit(1);
}
flag = 0;
}
}
// 最后一次未复制的复制过去
if (flag)
if (write(fd2,buf + start_offset,cnt - start_offset) == -1) {
std::cerr << "fd2 write failed\n";
exit(1);
}
out_offset += cnt;
}
return 0;
}
- 16 UNIX对目录树深度无要求,但是对路径长度有要求
PATH_MAX
限制,
1 |
|
本博客所有文章除特别声明外,均采用 CC BY-SA 4.0 协议 ,转载请注明出处!