• Linux dup和dup2


    Linux dup和dup2函数,他们有什么区别,什么场景下会用到,使用它们有什么注意事项

    dupdup2都是Linux系统中的系统调用,用于复制文件描述符。它们的主要区别在于如何指定新的文件描述符以及处理新文件描述符的方式。

    dup函数

    #include 
    int dup(int oldfd);
    
    • 1
    • 2

    dup函数创建一个新的文件描述符,这个新的文件描述符是当前可用文件描述符的最小数值。这个新的文件描述符和原来的文件描述符(oldfd)指向同一个文件,共享同一个数据结构,包括锁定、读写指针和各项权限或标志位。

    dup2函数

    #include 
    int dup2(int oldfd, int newfd);
    
    • 1
    • 2

    dup2函数与dup函数类似,但是它可以用参数newfd指定新文件描述符的数值。如果newfd已经被程序使用,则系统会先将newfd所指的文件关闭。如果newfd等于oldfd,则返回newfd,而不关闭newfd所指的文件。

    使用场景
    这两个函数通常用于重定向进程的stdin、stdout和stderr。例如,在CGI编程中,当需要将标准输入重定向到一个文件,或者将标准输出重定向到网络连接时,就可以通过dup()dup2()来实现。

    注意事项
    在使用这两个函数时,需要注意以下几点:

    1. 成功时,这两个函数都返回新的文件描述符。失败时,返回-1。
    2. 这两个函数返回的新文件描述符与原始文件描述符共享所有的锁定、读写位置和各项权限或flags等。
    3. 如果newfd已经打开,则dup2()会先关闭它。
    4. dup(oldfd)等效于fcntl(oldfd, F_DUPFD, 0),而dup2(oldfd, newfd)等效于先调用close(oldfd),然后调用fcntl(oldfd, F_DUPFD, newfd)

    关于dup和dup2的使用,复制新的文件描述符,对于旧的描述符的使用有什么影响,我的意思是,如果把复制出来的新的描述符关闭了,对于旧的文件描述符的行为是什么样的。还是说新的文件描述符关闭了,对旧的文件描述符没有任何关系,旧的文件描述符不想使用了还要关闭?

    当你使用dupdup2复制一个文件描述符时,新的文件描述符和旧的文件描述符都指向同一个打开的文件。这意味着它们共享相同的文件偏移量和状态标志。

    如果你关闭了新的文件描述符,这并不会影响到旧的文件描述符。旧的文件描述符仍然可以用来读写文件。这是因为在Linux系统中,关闭一个文件描述符只是减少了指向该打开文件的引用计数。只有当所有指向该打开文件的文件描述符都被关闭时,该打开的文件才会被真正地关闭。

    所以,如果你不再需要使用旧的文件描述符,你仍然需要显式地关闭它,即使你已经关闭了新的文件描述符。否则,旧的文件描述符将继续占用系统资源。


    如何使用dup和dup2函数

    dupdup2函数的使用主要涉及到文件描述符的复制。下面是一些示例代码来说明如何使用这两个函数。

    使用dup函数

    #include
    #include 
    #include 
    
    int main () {
        int file_desc = open("dup.txt", O_WRONLY | O_APPEND);
    
        if(file_desc < 0) 
            printf("Error opening the file\\n");
    
        int copy_desc = dup(file_desc);
    
        write(copy_desc,"This will be output to the file named dup.txt\\n", 46);
        write(file_desc,"This will also be output to the file named dup.txt\\n", 51);
    
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    在这个例子中,我们首先打开一个名为"dup.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup函数复制这个文件描述符,得到新的文件描述符copy_desc。最后我们分别使用file_desccopy_desc写入一些内容到同一个文件。

    使用dup2函数

    #include
    #include
    #include
    #include
    
    int main () {
        int file_desc = open("tricky.txt",O_WRONLY | O_APPEND);
    
        dup2(file_desc, 1);
    
        printf("I will be printed in the file tricky.txt\\n");
    
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    在这个例子中,我们首先打开一个名为"tricky.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到"tricky.txt"。这样,所有的printf语句都会被写入到"tricky.txt"文件中。


    举一个使用dup和dup2,当将标准输入重定向到一个文件,和重定向到进程的stdin、stdout和stderr,以及将标准输出重定向到网络连接例子

    将标准输入重定向到一个文件

    #include 
    #include 
    #include 
    
    int main() {
        int file_desc = open("input.txt", O_RDONLY);
        dup2(file_desc, 0);
    
        char buffer[1024];
        scanf("%s", buffer);
        printf("Read from input.txt: %s\n", buffer);
    
        close(file_desc);
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    在这个例子中,我们首先打开一个名为"input.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输入(文件描述符为0)重定向到"input.txt"。这样,所有的scanf语句都会从"input.txt"文件中读取数据。

    将标准输出重定向到一个文件

    #include 
    #include 
    #include 
    
    int main() {
        int file_desc = open("output.txt", O_WRONLY | O_CREAT, 0644);
        dup2(file_desc, 1);
    
        printf("This will be written to output.txt\n");
    
        close(file_desc);
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    在这个例子中,我们首先打开一个名为"output.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到"output.txt"。这样,所有的printf语句都会被写入到"output.txt"文件中。

    将标准输出重定向到网络连接
    这个例子涉及到网络编程,需要使用到socket编程。以下是一个简单的示例:

    #include 
    #include 
    #include 
    #include 
    #include 
    
    int main() {
        int sockfd = socket(AF_INET, SOCK_STREAM, 0);
        struct sockaddr_in servaddr;
    
        memset(&servaddr, 0, sizeof(servaddr));
        servaddr.sin_family = AF_INET;
        servaddr.sin_port = htons(22000);
        servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    
        connect(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr));
    
        dup2(sockfd, 1);
    
        printf("This will be sent to the network\n");
    
        close(sockfd);
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    在这个例子中,我们首先创建一个socket,并连接到本地的22000端口。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到这个socket。这样,所有的printf语句都会被发送到网络上。

    对于dup函数来说,新打开的文件描述符不想使用了,需要显示的关闭它是吧,那旧的描述符不想使用了是不是也要显示的关闭它呢?同理,对于dup2函数来说,新打开的文件描述符不想使用了,需要显示的关闭它是吧,那旧的描述符不想使用了是不是也要显示的关闭它呢?

    是的,你的理解是正确的。无论是dup函数还是dup2函数,当你不再需要使用新的文件描述符或旧的文件描述符时,你都需要显式地关闭它们。

    这是因为在Linux系统中,每个打开的文件都会占用系统资源,包括文件描述符。即使你已经使用dupdup2复制了文件描述符,原始的文件描述符和新的文件描述符都会继续占用系统资源,直到它们被显式地关闭。

    所以,如果你不再需要使用某个文件描述符,无论它是原始的还是复制出来的,你都应该使用close函数来关闭它。这样可以确保你的程序不会浪费系统资源。

    当一个进程结束时,操作系统会自动关闭该进程打开的所有文件描述符。所以,在这个特定的例子中,即使我们没有显式地关闭copy_descfile_desc,它们也会在程序退出时被自动关闭。

    但是,我要强调的是,即使操作系统会在进程结束时自动关闭所有打开的文件描述符,我们仍然应该养成在不再需要使用文件描述符时就立即关闭它们的习惯。这是因为如果一个长时间运行的程序忘记了关闭不再需要的文件描述符,那么这些文件描述符将继续占用系统资源,可能会导致资源泄漏。

  • 相关阅读:
    16. 从零开始编写一个类nginx工具, 反向代理upstream源码实现
    C语言课程设计学生考勤管理系统
    pinia状态管理器使用
    Airtest自定义启动器支持批量运行脚本,并兼容在AirtestIDE中使用
    MyBatis整合Spring的原理分析
    鸿蒙生态融合进行时!菊风启动适配HarmonyOS NEXT,赋能原生应用实时
    实操客群分层|无监督训练与有监督评估,面试中这两大类风控模型最会被问到的问题
    面试题:Redis缓存数据库,持久化机制有哪几种呢?
    谷粒学苑_第二天
    LeetCode 热题 HOT 100 第五十七天 221. 最大正方形 中等题 用python3求解
  • 原文地址:https://blog.csdn.net/u013318019/article/details/134008778