高级I/O函数(3)-tee、fcntl函数

时间:2021-07-08 20:56:44

tee函数使用

功能描述:tee函数在两个管道文件描述符之间复制数据,也是零拷贝操作.它不消耗数据,因此源文件描述符仍然可以用于后续的操作.

函数原型:

#include <fcntl.h>
ssize_t tee(int fd_in,int fd_out,size_t len,unsigned int flags);

函数参数:fd_in和fd_out必须都是管道文件描述符。

返回值:成功时返回在两个文件描述符之间复制的数据字节数,返回0表示没有复制任何数据.失败时返回-1并设置errno.

下面是利用tee函数和splice函数,实现同时输出数据到终端与文件的程序)的基本功能.

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <errno.h>
#include <fcntl.h> int main(int argc,const char* argv[]){
if(argc<=){
printf("usage:%s <file>\n",argv[]);
exit(-);
} int filefd=open(argv[],O_WRONLY|O_CREAT|O_TRUNC,);
assert(filefd>=); int pipefd_stdout[];
int ret=pipe(pipefd_stdout);
assert(ret!=-); int pipefd_file[];
ret=pipe(pipefd_file);
assert(ret!=-); /*将标准输入内容输入管道文件pipefd_stdout*/
ret=splice(STDIN_FILENO,NULL,pipefd_stdout[],NULL,,SPLICE_F_MORE
|SPLICE_F_MOVE);
assert(ret!=-); /*将管道pipefd_stdout[0]输出复制到管道pipefd_file的输入端*/
ret=tee(pipefd_stdout[],pipefd_file[],,SPLICE_F_NONBLOCK);
assert(ret!=-);
/*将管道pipefd_file的输出定向到文件描述符filefd上*/
ret=splice(pipefd_file[],NULL,filefd,NULL,,SPLICE_F_MORE|
SPLICE_F_MORE);
assert(ret!=-);
/*将管道pipefd_stdout的输出定向到标准输出,其内容与文件中完全一致*/
ret=splice(pipefd_stdout[],NULL,STDOUT_FILENO,NULL,,SPLICE_F_MORE|
SPLICE_F_MOVE);
assert(ret!=-); close(filefd);
close(pipefd_file[]);
close(pipefd_file[]);
close(pipefd_stdout[]);
close(pipefd_stdout[]);
return ;
}

fcntl函数的使用

功能描述:常用于控制文件描述符的属性和行为

函数原型:

#include <fcntl.h>
int fcntl(int fd,int cmd,...);

fd是被操作的文件描述符,cmd参数指定执行何种类型的操作.根据操作类型的不同,该函数可能还需要第三个可选参数.

fcntl支持的常用操作及其参数

操作分类 操作 含义 第三个参数的类型 成功时的返回值
复制文件描述符 F_DUPFD 创建一个新的文件描述符,其值大于或等于arg long 新创建的文件描述符的值
F_DUPFD_CLOEXEC 与F_DUPFD相似,不过在创建文件描述符的同时,设置其close_on_exec标志 long 新创建的文件描述符的值
获取和设置文件描述符的标志 F_GETFD 获取fd的标志 fd的标志
F_SETFD 设置fd的标志 long 0
获取和设置文件描述符的状态标志 F_GETFL 获取fd的状态标志,这些标志包括可由open系统调用设置的标志(O_CREAT、O_APPEND等)和访问模式(O_RDONLY、O_WRONLY和O_RDWR) void fd的状态标志
F_SETFL 设置fd的状态标志,但部分标志是不能修改的(比如访问模式标志) long 0
管理信号 F_GETOWN 获得SIGIO和SIGURG信号的宿主进程的PID或进程组的ID 信号的宿主进程或者进程组ID
F_SETOWN 设定SIGIO和SIGURG信号的宿主进程的PID和进程组的ID long  0
F_GETSIG 获取当应用程序被通知fd可读或可写时,是哪个信号通知该事件的 信号值
F_SETSIG 设置当fd可读或可写时,系统应该触发哪个信号来通知应用程序 long 0
操作管道容量 F_SETPIPE 设置由fd指定的管道的容量 long 0
F_GETPIPE 获取由fd指定的管道的容量 管道容量