【Linux】输入输出重定向

目录

一、概念

二、重定向的本质

三、系统调用接口dup和dup2

[3.1 dup](#3.1 dup)

[3.2 dup2](#3.2 dup2)


一、概念

在之前对Linux的学习中,我们有接触过一系列的重定向命令,例如 >、>>等

它们可以将一个命令的输出或输入重定向到其他地方,如echo命令用来将我们指定的文本打印到终端中,通过输出重定向就可以把原来要打印到终端的文本输出到其他的地方

例如我们可以通过重定向把echo要打印的内容重定向到某个文件中

像这样,通过控制数据的流向,让数据的目的地发生改变,即为 重定向


二、重定向的本质

在学习重定向之前,我们需要先知道什么是文件描述符

【Linux】文件描述符 fd-CSDN博客https://blog.csdn.net/Eristic0618/article/details/140838982?spm=1001.2014.3001.5501在前面的学习中,我们已经知道进程是通过文件描述符来访问文件的,而进程在打开一个文件时操作系统会给该文件分配一个当前最小的未使用的文件描述符。

所以假设我们把标准输出流关闭,那么1号文件描述符此时就是未被使用的状态

如果此时我们再打开一个文件,按理来说这个文件的文件描述符会分配为1

那么我们再向原来的标准输出流打印某些内容,这些内容会跑到哪里呢?

cpp 复制代码
#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
    close(1); //关闭标准输出
    int fd = open("file.txt", O_CREAT | O_WRONLY | O_TRUNC, 0666); //打开文件
    if(fd < 0)
    {
        perror("open");
        return 1;
    }
    const char *msg = "hello Linux\n";
    int cnt = 5;
    while(cnt--) //循环5次
    {
        write(1, msg, strlen(msg)); //向原来的标准输出流打印
    }
    close(fd);
    return 0;
}

原本当我们向write函数传入1号文件描述符进行写入时会向显示器文件中写入。但现在当我们向1号文件描述符中打印的时候,显示器上什么也没有,而我们写入的内容已经写到了file.txt中。此时,我们就完成了对这些内容的重定向了

所以可以知道:重定向实际上就是改变某个文件描述符指向的文件

原本1号文件描述符对应标准输出流,但我们通过关闭标准输出流再创建新文件的方式让1号描述符重新分配给了我们的file.txt,所以向显示器打印的内容就重定向到了我们的file.txt中

不过这种说法还不够准确,从内核的角度,应该说重定向就是修改文件描述符下标对应的file*指针


三、系统调用接口dup和dup2

dup和dup2都是Linux中用于实现重定向的系统调用接口,使用起来也并没有难度,不过二者之间有一些区别需要注意

3.1 dup

向dup中传入一个文件描述符,会给我们返回一个新的文件描述符,传入的文件描述符和新的文件描述符指向的是同一个文件

例如:

cpp 复制代码
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
    int newfd = dup(1);
    const char *msg = "hello Linux\n";
    write(newfd, msg, strlen(msg));
    printf("newfd: %d\n", newfd);
    return 0;
}

可以看到,我们向newfd中写入的内容被打印到了终端,说明newfd已经重定向到了标准输出中

3.2 dup2

dup2比dup多了一个参数,其中dup2会将newfd重定向到oldfd指向的文件,也就是将oldfd对应的file*指针拷贝到newfd对应的下标位置

cpp 复制代码
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
    int fd = open("file.txt", O_CREAT | O_WRONLY | O_TRUNC, 0666);
    dup2(fd, 1);
    const char *msg = "hello Linux\n";
    write(1, msg, strlen(msg));
    close(fd);
}

可以看到,因为1号文件描述符被重定向到了fd指向的文件,所以我们原本向标准输出中打印的内容并没有在终端中显示,而是被重定向到了file.txt中

除了对输出流进行重定向,我们也可以对输入流进行重定向,例如我们先向file.txt中写一些内容

然后用dup2将标准输入流重定向到文件对应的文件描述符

cpp 复制代码
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
    int fd = open("file.txt", O_RDONLY);
    dup2(fd, 0);
    char buffer[1024];
    ssize_t s = read(0, buffer, sizeof(buffer) - 1);
    if(s > 0)
    {
        buffer[s] = '\0';
        printf("%s", buffer);
    }
    close(fd);
}

执行程序,此时虽然从0号文件描述符中读取数据,但因为已经被重定向到了fd指向的文件中,所以read会从文件中读取数据,最后将读取的内容打印出来

如有错误欢迎在评论区指出

完.

相关推荐
远游客071321 分钟前
centos stream 8下载安装遇到的坑
linux·服务器·centos
马甲是掉不了一点的<.<21 分钟前
本地电脑使用命令行上传文件至远程服务器
linux·scp·cmd·远程文件上传
jingyu飞鸟22 分钟前
centos-stream9系统安装docker
linux·docker·centos
超爱吃士力架1 小时前
邀请逻辑
java·linux·后端
cominglately3 小时前
centos单机部署seata
linux·运维·centos
魏 无羡3 小时前
linux CentOS系统上卸载docker
linux·kubernetes·centos
CircleMouse3 小时前
Centos7, 使用yum工具,出现 Could not resolve host: mirrorlist.centos.org
linux·运维·服务器·centos
木子Linux4 小时前
【Linux打怪升级记 | 问题01】安装Linux系统忘记设置时区怎么办?3个方法教你回到东八区
linux·运维·服务器·centos·云计算
mit6.8244 小时前
Ubuntu 系统下性能剖析工具: perf
linux·运维·ubuntu
鹏大师运维4 小时前
聊聊开源的虚拟化平台--PVE
linux·开源·虚拟化·虚拟机·pve·存储·nfs