c语言 字符串dup,C语言中dup和dup2函数的不同和使用

在unix高级编程中有介绍dup和dup2,但是没有实例说明, 笔者自己结合实例进行了测试了解。

在linux下,通过open打开以文件后,会返回一个文件描述符,文件描述符会指向一个文件表,文件表中的节点指针会指向节点表。看下图:

f78b514b29f2313344ff084b09ee709b.png

dup和dup2两个函数都可以用来复制打开的文件描述符,复制成功后和复制源共享同一个文件表。看下图:

48229189a2339914981f7f325252705b.png

dup函数

dup(现存的文件描述符)

dup返回的新文件描述符一定是当前可以用描述符中的最小值。下面先打开一个文件来看下文件描述符,为保证测试成功,创建一个测试文件log.txt。

1

2

3

4

5

6

7

8

9

10

11

12

#include

#include

int main(int argc,char *argv[])

{

int fd;

fd = open("./log.txt", O_RDWR);

printf("%d\n", fd);

return 0;

}

上面的代码用读写打开了log.txt这个文件,编译上面的代码然后执行,执行成功的话,应当是输出3,因为0,1,2分别被标准输入,标准输出,标准错误输出占用了。使用dup复制这个文件描述符,并尝试移动fd偏移量:

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

#include

#include

#include

int main(int argc,char *argv[])

{

int fd, copyfd;

fd = open("./log.txt", O_RDWR);

//复制fd

copyfd = dup(fd);

//输出copyfd,应当为4

printf("%d\n", copyfd);

//打印出fd和copyfd的偏移量,都为0

printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));

printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));

//将fd的偏移量+3

lseek(fd, 3, SEEK_SET);

//打印出fd和copyfd的偏移量,都为3

printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));

printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));

return 0;

}

编译执行上例代码可以发现当移动fd的偏移量时,copyfd的偏移量也发生了变化。往文件里写入内容试试,先把log.txt内容清空。

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

#include

#include

#include

int main(int argc,char *argv[])

{

int fd, copyfd;

fd = open("./log.txt", O_RDWR);

//复制fd

copyfd = dup(fd);

char buf1[] ="hello ";

char buf2[] ="world!";

//往fd文件写入内容

if (write(fd, buf1, 6) != 6) {

printf("write error!");

}

//打印出fd和copyfd的偏移量,经过上面的写操作,都变成6了

printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));

printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));

//往copyfd写入内容

if (write(copyfd, buf2, 6) != 6) {

printf("write error!");

}

return 0;

}

编译执行程序,log.txt的就有hello world!字符串了。

dup2函数

dup2(现存的文件描述符,可用的文件描述符)

dup2和dup函数一样,只是返回的文件描述符可以通过第二个参数”可用的文件描述符“指定。如果“可用的文件描述符“是打开状态,则会被关闭;如果”现存的文件描述符“和”可用的文件描述符“一样,则不会关闭,笔者认为这两个参数值一样的话,代码是没有任何意义的。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

#include

#include

#include

int main(int argc,char *argv[])

{

int fd, copyfd;

fd = open("./log.txt", O_RDWR);

//指定文件描述符号为1000

copyfd = dup2(fd, 1000);

//打印fd和copyfd,应当输出3 1000

printf("%d %d\n", fd, copyfd);

return 0;

}

上面程序就是指定返回的文件描述为1000,再来看下指定的文件描述符是打开的情况,修改上例代码,将文件描述符指定为1:

1

copyfd = dup2(fd, 1);

编译执行程序将看不到任何输出,因为1是终端标准输出的标识符,经过这样一复制后,标准输出就被关闭了,使用printf自然看不到输出信息了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的C语言实现Shell脚本重定向函数check4redirection和执行函数execute的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <unistd.h> #define MAXARGS 20 #define MAXLINE 80 int check4redirection(char *cmd, int *in, int *out) { int i, len, pos = 0; char *args[MAXARGS]; char *infile, *outfile; len = strlen(cmd); for (i = 0; i < len; i++) { if (cmd[i] == '<') { infile = strtok(&cmd[i+1], " \t\n"); *in = open(infile, O_RDONLY); if (*in == -1) { perror("Open file error"); return -1; } cmd[i] = '\0'; } if (cmd[i] == '>') { outfile = strtok(&cmd[i+1], " \t\n"); *out = open(outfile, O_WRONLY|O_TRUNC|O_CREAT, 0644); if (*out == -1) { perror("Open file error"); return -1; } cmd[i] = '\0'; } } args[pos++] = strtok(cmd, " \t\n"); while ((args[pos++] = strtok(NULL, " \t\n")) != NULL); args[pos] = NULL; if (*in != STDIN_FILENO) { if (dup2(*in, STDIN_FILENO) == -1) { perror("Dup2 error"); return -1; } close(*in); } if (*out != STDOUT_FILENO) { if (dup2(*out, STDOUT_FILENO) == -1) { perror("Dup2 error"); return -1; } close(*out); } return execvp(args[0], args); } int execute(char *cmd) { int pid, status, in = STDIN_FILENO, out = STDOUT_FILENO; if ((pid = fork()) == -1) { perror("Fork error"); return -1; } else if (pid == 0) { return check4redirection(cmd, &in, &out); } else { while (wait(&status) != pid); return status; } } int main(int argc, char *argv[]) { char line[MAXLINE]; int status; while (1) { printf("myShell> "); fgets(line, MAXLINE, stdin); if (strcmp(line, "exit\n") == 0) exit(0); status = execute(line); printf("Exit status: %d\n", status); } return 0; } ``` 检查重定向的函数check4redirection接受一个命令字符串,以及两个指针(in和out),用于存储输入和输出的文件描述符。该函数首先使用strtok函数将命令字符串拆分为命令和参数数组args,然后在命令字符串获取输入和输出文件名,并打开相应文件。最后,该函数使用dup2函数将输入和输出文件描述符复制到STDIN_FILENO和STDOUT_FILENO,并关闭原始文件描述符。最后,该函数使用execvp函数执行指定的命令和参数。 执行函数execute接受一个命令字符串并在新进程调用check4redirection函数。如果fork失败,则返回-1。在子进程,调用check4redirection函数执行命令。在父进程,等待子进程执行完毕,并返回子进程的退出状态。主函数使用fgets从标准输入读取命令行,并在循环执行命令。如果用户输入exit,则退出程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值