浅谈 dup 和 dup2 的用法

一、dup 和 dup2 函数

这两个函数都可以来复制一个现有的文件描述符,他们的声明如下:

#include <unistd.h>
int dup(int fd);
int dup2(int fd, int fd2);

关于 dup 函数,当我们调用它的时候,dup 会返回一个新的描述符,这个描述一定是当前可用文件描述符中的最小值。

我们知道,一般的 0、1、2 描述符分别被标准输入、输出、错误占用,所以在程序中如果 close 掉标准输出 1 后,调用 dup 函数,此时返回的描述符就是 1 。

对于 dup2,可以用 fd2 指定新描述符的值,如果 fd2 本身已经打开了,则会先将其关闭。如果 fd 等于 fd2,则返回 fd2,并不关闭它。

这两个函数返回的描述符与 fd 描述符所指向的文件共享同一文件表项。如下图所示: 

这里写图片描述

也就是 fd 与 fd2 可对同一个文件进行读写操作。且其是一种原子操作。

二、重定向示例

1、duptest.c

#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <unistd.h>
#include <string.h>

int main(int argc, char *argv[])
{
    int i_fd = open("hello.txt", O_CREAT | O_APPEND | O_RDWR, 0666);

    if (i_fd < 0) {
        printf("open error!\n");
        return 0;
    }
    const char *buf = "hello fd\n";
    int bufLen = strlen(buf);
    if (write(i_fd, buf, bufLen) != bufLen) {
        printf("write fd error\n");
    }
    printf("write fd char len is %d\n", bufLen);

    int i_dup_fd = dup(i_fd);
    if (i_dup_fd < 0) {
        printf("dup error!\n");
        return 0;
    }

    printf("i_dup_fd = %d \t i_fd = %d\n", i_dup_fd, i_fd);
    close(i_fd);

    const int bufLen1 = 10;
    char c_buffer[bufLen1];
    int n = 0;
    while ((n = read(STDIN_FILENO, c_buffer, bufLen1)) != 0) {
        printf("write dup fd char count is %d!\n", n);
        if (write(i_dup_fd, c_buffer, n) != n) {
            printf("write dup fd error!\n");
            return 0;
        }
    }
    return 0;
}

如上打开一个文件,我们先写入文件内容“hello fd”,然后将 fd 的描述符拷贝到 dup_fd 的文件描述符上,然后将标准输入的内容写入到 dup_fd 的文件中。

运行程序如下:

这里写图片描述

 查看文件可以看到文件内容如下:

这里写图片描述

 2、dup2test.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>

int main(int argc, char *argv[])
{
    int i_fd = open("hello_dup2.txt", O_CREAT | O_APPEND | O_RDWR, 0666);

    if (i_fd < 0) {
        printf("open error!\n");
        return 0;
    }
    const char *buf = "hello i_fd\n";
    const int bufLen = strlen(buf);
    if (write(i_fd, buf, bufLen) != bufLen) {
        printf("write dup2 error\n");
    }

    int i_dup2_fd = dup2(i_fd, STDOUT_FILENO);
    if (i_dup2_fd != STDOUT_FILENO) {
        printf("error dup2!\n");
        return 0;
    }
    close(i_fd);

    char c_buf[1024];
    int i_read_n = 0;
    while ((i_read_n = read(STDIN_FILENO, c_buf, 1024)) != 0) {
        i_read_n = read(STDIN_FILENO, c_buf + i_read_n, sizeof(c_buf) - 1 - i_read_n);

        if (i_read_n < 0) {
            printf("read error!\n");
            return 0;
        }

        printf("%s", c_buf);
        fflush(stdout);
        sleep(1);
    }
    close(i_dup2_fd);

    return 0;
}

如上,这里没有像使用 dup 的时候显示的调用 write 函数将标准输入的内容写入到指定文件中,而是将标准输出重定向到指定文件中,然后调用 printf 函数将标准输出的内容重定向到指定文件中。

我们在写简单的日志时就可以将 printf 的内容重定向到日志中,使用 printf 作为写日志的接口。

如上运行程序如下:

这里写图片描述

查看 hello_dup2.txt 可以看到如下:

这里写图片描述

转载:浅谈dup和dup2的用法_非正经程序员的博客-CSDN博客_dup2 

(SAW:Game Over!) 

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
dupdup2是Unix和Linux系统中的系统调用函数,用于复制文件描述符。 dup函数的原型为: ```c int dup(int oldfd); ``` dup2函数的原型为: ```c int dup2(int oldfd, int newfd); ``` 这两个函数的作用是将一个已有的文件描述符复制到另一个文件描述符中。 - dup函数会返回一个新的文件描述符,该描述符与参数oldfd指向同一文件表项。这意味着对返回的文件描述符的读写操作会影响到原来的文件描述符。 - dup2函数也会返回一个新的文件描述符,但是它可以指定新的文件描述符newfd。如果newfd已经被使用,那么dup2函数会先将newfd关闭,然后再复制oldfd,使得新的文件描述符与oldfd指向同一文件表项。 这两个函数通常用于重定向标准输入、输出和错误流。例如,可以使用dup2函数将标准输出重定向到一个文件中: ```c #include <unistd.h> #include <fcntl.h> int main() { int fd = open("output.txt", O_WRONLY | O_CREAT, 0644); if (fd == -1) { perror("open"); return 1; } // 将标准输出重定向到文件 if (dup2(fd, STDOUT_FILENO) == -1) { perror("dup2"); return 1; } // 打印到标准输出,实际上会写入文件 printf("Hello, world!\n"); close(fd); return 0; } ``` 在上面的示例中,dup2函数将文件描述符fd复制到标准输出描述符STDOUT_FILENO中,这样所有的printf函数调用都会将内容写入文件"output.txt"中。 希望这个解答能够帮到你!如果还有其他问题,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值