重定向编程 dup和dup2函数

本文详细介绍了dup和dup2函数在Unix/Linux系统中如何用于文件描述符的复制,解释了它们在重定向标准输入输出中的作用。通过具体实例展示了如何使用这两个函数来复制文件描述符,并将标准输出重定向到指定文件。

重定向编程 dup和dup2函数
函数dup和dup2提供了复制文件描述符的功能。他们通常用于stdin,stdout或进程的stderr的重定向。一般来说,普通输出函数(如:printf),默认是将某信息写入到文件描述符为1的文件中,普通输入函数都默认从文件描述符为0的文件中读取数据。因此重定向操作实际上是关闭某个标准输入输出设备(文件描述符为0、1、2),而将另一个打开的普通文件的文件描述符设置为0、1、2.
输入重定向:关闭标准输入设备,打开(或复制)某普通文件,使其文件描述符为0.
输出重定向:关闭标准输出设备,打开(或复制)某普通文件,使其文件描述符为1.
错误输出重定向:关闭标准错误输入设备,打开(或复制)某普通文件,使其文件描述符为2.
1.dup函数
头文件:#include<unistd.h>
定义函数:
int dup(int oldfd);
函数说明:
函数dup 允许你复制一个oldfd文件描述符。存入一个已存在的文件描述符,它就会返回一个与该描述符“相同”的新的文件描述符。即这两个描述符共享相同的内部结构,共享所有的锁定,读写位置和各项权限或flags等等。例如:对一个文件描述符进行了lseek操作,另一个文件描述符的读写位置也会随之改变。不过,文件描述符之间并不共享close-on-exec flags.
返回值:如成功则返回新的文件描述符,否则出错返回-1.注意:由dup函数返回的新文件描述符一定是当前可用文件描述符中的最小值。

EBADF:参数fd非有效的文件描述符,或该文件已关闭。
假设进程执行了:newfd = dup(1);
当此函数执行时,假定下一个可用的文件描述符是3.因为两个描述符指向同一个文件表项,所以他们共享同一文件状态标志以及同一当前文件偏移量。如图1所示。

在这里插入图片描述
图1 复制文件描述符的内核数据结构

例子1:复制文件描述符,并向文件写数据.

#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
 
void main()
{
        int fd,newfd;
        char *bufFD="Advanced Programming! write by fd\n";
        char *bufNewFD="Advanced Programming! write by NewFD\n";
        fd = open("test.txt",O_RDWR|O_CREAT,0644);
        if(fd==-1)
        {
                printf("open file error%m\n");
                exit(-1);
        }
 
        //开始复制了
        newfd = dup(fd);
        //使用fd写
        write(fd,bufFD,strlen(bufFD));
        close(fd);
 
        //使用newfd写
        write(newfd,bufNewFD,strlen(bufNewFD));
 
        if(close(newfd)==-1)
        {
                printf("close error\n");
        }
        exit(0);
}

输出:
查看test.txt文件,里面有两段字符串:
:Advanced Programming! write by fd
:Advanced Programming! write by newfd
可以看出,对fd或newfd进行读写操作时对同一个文件操作,而且还可以看到fd关闭后,对newfd没有影响,使用newfd还可以操作打开的文件。
2.dup2函数
头文件:#include<unistd.h>
定义函数:int dup2(int oldfd, int newfd);
函数说明:
dup2用来复制参数oldfd所指的文件描述符,并将oldfd拷贝到参数newfd后一起返回。若参数newfd为一个打开的文件描述符,则newfd所指的文件会先被关闭,若newfd等于oldfd,则返回newfd,而不关闭newfd所指的文件。dup2所复制的文件描述符与原来的文件描述符共享各种文件状态。共享所有的锁定,读写位置和各项权限或flags等等.
返回值:如成功则返回新的文件描述符,否则出错返回-1. 注意:由dup2函数返回的新文件描述符一定是当前可用文件描述符中的最小值。
newfd和oldfd具有的共同点:
(1)相同的打开文件(管道)。
(2)相同的文件指针,即两个文件共享一个文件指针。
(3)相同的访问模式。读取、写入。
(4)相同的文件状态标识。
错误代码:
EBADF:参数fd非有效的文件描述符,或该文件已关闭。
例子2:将标准输出重定向到目标文件test.txt

#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
 
void main()
{
        int fd;
        int refd;
        char *buf="Advanced Programming!\n";
        fd = open("test.txt",O_RDWR|O_CREAT,0644);
        if(fd==-1)
        {
                printf("open file error:%m\n");
                exit(-1);
        }
        refd = dup2(fd,fileno(stdout));
        if(refd==-1)
        {
                printf("redirect standard out error:%m\n");
                exit(-1);
        }
 
        //写数据,本应该写入到stdout的信息重定向而写入到目标文件中(test.txt)
        printf("dup2的返回值:%d\n",refd);
        write(fileno(stdout),buf,strlen(buf));
        close(fd);
        exit(0);
}

输出:
查看test.txt文件,可以看到以下内容
:Advanced Programming!
:dup2的返回值:1
printf(“dup2的返回值:%d\n”,refd);write(fileno(stdout),buf,strlen(buf));
写数据,本应该写入到stdout的信息,但是标准输出已经重定向到目标文件中,故向标准输出写的数据将会写到目标文件中。读者看以尝试输出fd的前后的值?


原文:https://blog.youkuaiyun.com/u010006102/article/details/39667875

### C语言 `dup2` 文件描述符重定向 在C语言中,`dup2()` 系统调用用于复制一个已有的文件描述符并将其绑定到指定的目标文件描述符上。这通常用来实现输入/输出流的重定向。 #### 头文件包含 为了使用 `dup2()` 其他必要的I/O操作函数,程序需要包含如下头文件: ```c #include <unistd.h> #include <fcntl.h> ``` #### 函数原型 `int dup2(int oldfd, int newfd);` - 参数 `oldfd`: 被复制的源文件描述符。 - 参数 `newfd`: 新创建的目的文件描述符;如果此位置已有打开的文件,则会先关闭它再执行复制动作[^4]。 #### 实现示例 下面给出一段完整的代码来展示如何利用 `dup2()` 来改变标准输出的方向至一个新的文件: ```c #include <stdio.h> #include <unistd.h> #include <fcntl.h> int main() { // 打开目标文件 "output.txt" int fd = open("output.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644); if (fd == -1) { perror("Failed to open file"); return 1; } // 将标准输出重定向到新打开的文件描述符 'fd' if (dup2(fd, STDOUT_FILENO) == -1) { perror("Failed to redirect stdout"); return 1; } // 关闭原始文件描述符以防资源泄露 close(fd); // 测试打印语句会被写入 output.txt 中而不是控制台终端 printf("This message goes into the file instead of terminal.\n"); return 0; } ``` 这段代码首先尝试以可写模式打开名为 `"output.txt"` 的文件,并获取其对应的文件描述符 `fd`. 接着通过调用 `dup2()` 方法把当前的标准输出(即 `STDOUT_FILENO`, 它代表的是文件描述符1)指向新的文件描述符 `fd` 。最后一步是测试性的输出一条消息给用户,但由于之前已经改变了默认输出方向,因此这条信息将会被记录到磁盘上的文件里而非显示于屏幕之上[^3]. #### 注意事项 当成功完成上述过程之后,任何后续针对标准输出的操作都将影响到所选中的外部文件,直到再次更改为止。另外需要注意的一点是,在实际应用当中应当始终记得释放不再使用的文件描述符以免造成不必要的内存泄漏问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值