Linux系统编程——文件描述符的复制:dup()和dup2()

dup() 和 dup2() 是两个非常有用的系统调用,都是用来复制一个文件的描述符,使新的文件描述符也标识旧的文件描述符所标识的文件。

这个过程类似于现实生活中的配钥匙,钥匙相当于文件描述符,锁相当于文件,本来一个钥匙开一把锁,相当于,一个文件描述符对应一个文件,现在,我们去配钥匙,通过旧的钥匙复制了一把新的钥匙,这样的话,旧的钥匙和新的钥匙都能开启这把锁。对比于
dup(), dup2() 也一样,通过原来的文件描述符复制出一个新的文件描述符,这样的话,原来的文件描述符和新的文件描述符都指向同一个文件,我们操作这两个文件描述符的任何一个,都能操作它所对应的文件。

所需头文件:

#include <unistd.h>

int dup(int oldfd);

功能:

通过 oldfd 复制出一个新的文件描述符,新的文件描述符是调用进程文件描述符表中最小可用的文件描述符,最终 oldfd 和新的文件描述符都指向同一个文件。

参数:

oldfd: 需要复制的文件描述符 oldfd

返回值:

成功:新文件描述符

失败:-1

下面的例子为,打开一个文件,复制文件描述符,通过 2 个描述符分别对文件进行写操作:

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

int main(int argc, char *argv[])
{
	int fd1;
	int fd2;

	// 打开文件
	fd1 = open("1.txt", O_CREAT|O_WRONLY, 0666);
	if (fd1 < 0){
		perror("open");
		exit(-1);
	}
	printf("fd1 ============== %d\n", fd1);

	// 通过 fd1 复制出 fd2, 最终 fd1, fd2 都指向 1.txt
	fd2 = dup(fd1);
	printf("fd2 ============== %d\n", fd2);

	char *buf1 = "this is a test for fd1\n";
	// 操作 fd1 文件描述符
	write(fd1, buf1, strlen(buf1));

	char *buf2 = "this is a test for fd2\n";
	// 操作 fd2 文件描述符
	write(fd2, buf2, strlen(buf2));

	// 关闭文件描述符,两个都得关
	close(fd1);
	close(fd2);

	return 0;
}

运行结果如下:

通过上面的运行结果得知,dup() 后复制的新文件描述符是系统自动分配的最小可用的文件描述符。

接下来,我们再写一个例子,把本来通过 printf() 显示到屏幕上的内容,不显示在屏幕上,而让这些内容写入一个文件里:

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

int main(int argc, char *argv[])
{
	int fd1;
	int fd2;

	// 打开文件
	fd1 = open("1.txt", O_CREAT|O_WRONLY, 0666);
	if (fd1 < 0){
		perror("open");
		exit(-1);
	}
	printf("fd1 ============== %d\n", fd1);

	// 1 本来指向标准输出设备(如,显示屏)
	// 把 1 文件描述符关闭,就是说 1 不再指向标准输出设备
	// 这样的话,1 文件描述符就空闲了,它就成为最小可用的文件描述符
	close(1);

	// 通过 fd1 复制出 fd2, 最终 fd1, fd2 都指向 “1.txt”
	// 系统会给 fd2 分配一个最小可用的文件描述符 1,即 fd2 = 1
	fd2 = dup(fd1);

	// 下面这句话的内容不会打印到屏幕上,而会写到文件 “1.txt” 里
	// printf()是标准库函数,最终还是会调用系统调用函数write()
	// 相当于这样,write(1,),往 1 文件描述符写内容,
	// 默认的情况下, 1 文件描述符指向标准输出设备(如,显示屏)
	// 所以, printf()的内容先显示到屏幕上
	// 但是现在 1 文件描述符指向文件 “1.txt”
	// 所以,printf()的内容会写入文件 “1.txt”
	printf("fd2 ============== %d\n", fd2);

	close(fd1);
	close(fd2);

	return 0;
}

运行结果如下:

接下来,我们继续一起学习 dup2() 的用法,功能和 dup() 完全一样,但是 dup2() 复制出来的新文件描述符可以指定任意一个合法的数字。

int dup2(int oldfd, int newfd)

功能:

通过 oldfd 复制出一个新的文件描述符 newfd,如果成功,newfd 和函数返回值是同一个返回值,最终 oldfd 和新的文件描述符 newfd 都指向同一个文件。

参数:

oldfd: 需要复制的文件描述符

newfd: 新的文件描述符,这个描述符可以人为指定一个合法数字(0-1023),如果指定的数子已经被占用(和某个文件有关联),此函数会自动关闭 close() 断开这个数字和某个文件的关联,再来使用这个合法数字。

返回值:

成功:返回 newfd

失败:返回 -1

接着,我们将上面的例子改为用 dup2() 来实现:

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

int main(int argc, char *argv[])
{
	int fd1;
	int fd2;

	// 打开文件
	fd1 = open("1.txt", O_CREAT|O_WRONLY, 0666);
	if (fd1 < 0){
		perror("open");
		exit(-1);
	}
	printf("fd1 ============== %d\n", fd1);

	// 通过 fd1 复制出 fd2, 最终 fd1, fd2 都指向 “1.txt”
	// 指定 fd2 的值为 1,1 原来指向标准输出设备,先 close(),再复制
	fd2 = dup2(fd1, 1);

	// 下面这句话的内容不会打印到屏幕上,而会写到文件 “1.txt” 里
	// printf()是标准库函数,最终还是会调用系统调用函数write()
	// 相当于这样,write(1,),往 1 文件描述符写内容,
	// 默认的情况下, 1 文件描述符指向标准输出设备(如,显示屏)
	// 所以, printf()的内容先显示到屏幕上
	// 但是现在 1 文件描述符指向文件 “1.txt”
	// 所以,printf()的内容会写入文件 “1.txt”
	printf("fd2 ============== %d\n", fd2);

	close(fd1);
	close(fd2);

	return 0;
}

运行结果如下:

本教程示例代码下载请点此处。

时间: 2024-10-10 02:36:18

Linux系统编程——文件描述符的复制:dup()和dup2()的相关文章

Linux系统编程--文件描述符的复制dup()和dup2()【转】

本文转载自:http://blog.csdn.net/tennysonsky/article/details/45870459 dup() 和 dup2() 是两个非常有用的系统调用,都是用来复制一个文件的描述符,使新的文件描述符也标识旧的文件描述符所标识的文件. 这个过程类似于现实生活中的配钥匙,钥匙相当于文件描述符,锁相当于文件,本来一个钥匙开一把锁,相当于,一个文件描述符对应一个文件,现在,我们去配钥匙,通过旧的钥匙复制了一把新的钥匙,这样的话,旧的钥匙和新的钥匙都能开启这把锁.对比于 d

文件描述符的复制——实现输出重定向

dup(2)   dup2(2) #include <unistd.h>int dup(int oldfd);功能:复制文件描述符参数:oldfd:指定源文件的描述符返回值:错误 -1 errno被设置成功 返回新的文件描述符 新的文件描述符使用未使用的.最小的文件描述符 int dup2(int oldfd, int newfd);功能:复制文件描述符参数:oldfd:源描述符newfd:目的描述符返回值:错误 -1 errno被设置成功 返回新的文件描述符 1 //利用dup和dup2完成

Linux下基于文件描述符的文件操作(非缓冲)

1 文件描述符 内核为每个进程维护一个已打开文件的记录表(实现为结构体数组),文件描述符是一个较小的正整数(0-1023)(结构体数组下标),它代表记录表的一项,通过文件描述符和一组基于文件描述符的文件操作函数,就可以实现对文件的读.写.创建.删除等操作. 常用基于文件描述符的函数有open(打开).creat(创建).close(关闭).read(读取).write(写入).ftruncate(改变文件大小).lseek(定位).fsync(同步).fstat(获取文件状态).fchmod(权

【Linux驱动】文件描述符以及相关知识

1.文件描述符 Linux操作系统中,几乎所有的设备都被抽象成为设备文件.因此,当我们想对设备进行操作的时候可以直接去操作其相应的设备文件.设备文件即是文件,要想对文件进行操作,无非就是:打开文件.关闭文件.写入数据.读出数据等,它们分别对应的函数有open(),close(),write(),read(),就以其中的open()函数做一个分析.open函数的作用是打开一个文件. (1)它的定义:int open( const char * pathname, int flags); int o

【翻译自mos文章】在unix/linux中使用文件描述符(File Descriptors)来找回被删掉的文件(数据文件or redo log)

在unix/linux中使用文件描述符(File Descriptors)来找回被删掉的文件(数据文件or redo log) 参考原文: Retrieve deleted files on Unix / Linux using File Descriptors (Doc ID 444749.1) 适用于: Oracle Database - Enterprise Edition - Version 8.1.7.0 to 11.2.0.3 [Release 8.1.7 to 11.2] Linu

每天进步一点点——Linux中的文件描述符与打开文件之间的关系

转载请说明出处:http://blog.csdn.net/cywosp/article/details/38965239 1. 概述 在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件.目录文件.链接文件和设备文件.文件描述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符.程序刚刚启动的时候,0是标准输入,1是标准输出,2是标准错误.如果此时去打开一个

(转)Linux中的文件描述符

本文转自:http://blog.csdn.net/cywosp/article/details/38965239 作者:cywosp 1. 概述 在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件.目录文件.链接文件和设备文件.文件描述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符.程序刚刚启动的时候,0是标准输入,1是标准输出,2是标准错误.如果

Linux中的文件描述符与打开文件之间的关系

1. 概述 在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件.目录文件.链接文件和设备文件.文件描述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符.程序刚刚启动的时候,0是标准输入,1是标准输出,2是标准错误.如果此时去打开一个新的文件,它的文件描述符会是3.POSIX标准要求每次打开文件时(含socket)必须使用当前进程中最小可用的文件描述符号

Linux系统编程---文件I/O思想

1.文件如何在Linux中存储 Linux中任何事物都可以用一个文件来表示,或者通过特殊的文件提供. 一个文件由目录项.inode和数据块组成. 硬盘的最小存储单位叫做"扇区"(Sector),每个扇区储存512字节(相当于0.5KB). 操作系统读取硬盘的时候,不会一个扇区一个扇区地读取,这样效率太低,而是一次性连续读取多个扇区,即一次性读取一个"块"(block).这种由多个扇区组成的"块",是文件存取的最小单位."块"的