Linux dup和dup2
Linux dup和dup2函数,他们有什么区别,什么场景下会用到,使用它们有什么注意事项
dup和dup2都是Linux系统中的系统调用,用于复制文件描述符。它们的主要区别在于如何指定新的文件描述符以及处理新文件描述符的方式。
dup函数
#include <unistd.h>
int dup(int oldfd);
dup函数创建一个新的文件描述符,这个新的文件描述符是当前可用文件描述符的最小数值。这个新的文件描述符和原来的文件描述符(oldfd)指向同一个文件,共享同一个数据结构,包括锁定、读写指针和各项权限或标志位。
dup2函数
#include <unistd.h>
int dup2(int oldfd, int newfd);
dup2函数与dup函数类似,但是它可以用参数newfd指定新文件描述符的数值。如果newfd已经被程序使用,则系统会先将newfd所指的文件关闭。如果newfd等于oldfd,则返回newfd,而不关闭newfd所指的文件。
使用场景
这两个函数通常用于重定向进程的stdin、stdout和stderr。例如,在CGI编程中,当需要将标准输入重定向到一个文件,或者将标准输出重定向到网络连接时,就可以通过dup()和dup2()来实现。
注意事项
在使用这两个函数时,需要注意以下几点:
- 成功时,这两个函数都返回新的文件描述符。失败时,返回-1。
- 这两个函数返回的新文件描述符与原始文件描述符共享所有的锁定、读写位置和各项权限或flags等。
- 如果newfd已经打开,则
dup2()会先关闭它。 dup(oldfd)等效于fcntl(oldfd, F_DUPFD, 0),而dup2(oldfd, newfd)等效于先调用close(oldfd),然后调用fcntl(oldfd, F_DUPFD, newfd)。
关于dup和dup2的使用,复制新的文件描述符,对于旧的描述符的使用有什么影响,我的意思是,如果把复制出来的新的描述符关闭了,对于旧的文件描述符的行为是什么样的。还是说新的文件描述符关闭了,对旧的文件描述符没有任何关系,旧的文件描述符不想使用了还要关闭?
当你使用dup或dup2复制一个文件描述符时,新的文件描述符和旧的文件描述符都指向同一个打开的文件。这意味着它们共享相同的文件偏移量和状态标志。
如果你关闭了新的文件描述符,这并不会影响到旧的文件描述符。旧的文件描述符仍然可以用来读写文件。这是因为在Linux系统中,关闭一个文件描述符只是减少了指向该打开文件的引用计数。只有当所有指向该打开文件的文件描述符都被关闭时,该打开的文件才会被真正地关闭。
所以,如果你不再需要使用旧的文件描述符,你仍然需要显式地关闭它,即使你已经关闭了新的文件描述符。否则,旧的文件描述符将继续占用系统资源。
如何使用dup和dup2函数
dup和dup2函数的使用主要涉及到文件描述符的复制。下面是一些示例代码来说明如何使用这两个函数。
使用dup函数
#include<stdio.h>
#include <unistd.h>
#include <fcntl.h>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;
}
在这个例子中,我们首先打开一个名为"dup.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup函数复制这个文件描述符,得到新的文件描述符copy_desc。最后我们分别使用file_desc和copy_desc写入一些内容到同一个文件。
使用dup2函数
#include<stdlib.h>
#include<unistd.h>
#include<stdio.h>
#include<fcntl.h>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;
}
在这个例子中,我们首先打开一个名为"tricky.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到"tricky.txt"。这样,所有的printf语句都会被写入到"tricky.txt"文件中。
举一个使用dup和dup2,当将标准输入重定向到一个文件,和重定向到进程的stdin、stdout和stderr,以及将标准输出重定向到网络连接例子
将标准输入重定向到一个文件
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>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;
}
在这个例子中,我们首先打开一个名为"input.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输入(文件描述符为0)重定向到"input.txt"。这样,所有的scanf语句都会从"input.txt"文件中读取数据。
将标准输出重定向到一个文件
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>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;
}
在这个例子中,我们首先打开一个名为"output.txt"的文件,并获取其文件描述符file_desc。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到"output.txt"。这样,所有的printf语句都会被写入到"output.txt"文件中。
将标准输出重定向到网络连接
这个例子涉及到网络编程,需要使用到socket编程。以下是一个简单的示例:
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>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;
}
在这个例子中,我们首先创建一个socket,并连接到本地的22000端口。然后我们使用dup2函数将标准输出(文件描述符为1)重定向到这个socket。这样,所有的printf语句都会被发送到网络上。
对于dup函数来说,新打开的文件描述符不想使用了,需要显示的关闭它是吧,那旧的描述符不想使用了是不是也要显示的关闭它呢?同理,对于dup2函数来说,新打开的文件描述符不想使用了,需要显示的关闭它是吧,那旧的描述符不想使用了是不是也要显示的关闭它呢?
是的,你的理解是正确的。无论是dup函数还是dup2函数,当你不再需要使用新的文件描述符或旧的文件描述符时,你都需要显式地关闭它们。
这是因为在Linux系统中,每个打开的文件都会占用系统资源,包括文件描述符。即使你已经使用dup或dup2复制了文件描述符,原始的文件描述符和新的文件描述符都会继续占用系统资源,直到它们被显式地关闭。
所以,如果你不再需要使用某个文件描述符,无论它是原始的还是复制出来的,你都应该使用close函数来关闭它。这样可以确保你的程序不会浪费系统资源。
当一个进程结束时,操作系统会自动关闭该进程打开的所有文件描述符。所以,在这个特定的例子中,即使我们没有显式地关闭copy_desc和file_desc,它们也会在程序退出时被自动关闭。
但是,我要强调的是,即使操作系统会在进程结束时自动关闭所有打开的文件描述符,我们仍然应该养成在不再需要使用文件描述符时就立即关闭它们的习惯。这是因为如果一个长时间运行的程序忘记了关闭不再需要的文件描述符,那么这些文件描述符将继续占用系统资源,可能会导致资源泄漏。
相关文章:
Linux dup和dup2
Linux dup和dup2函数,他们有什么区别,什么场景下会用到,使用它们有什么注意事项 dup和dup2都是Linux系统中的系统调用,用于复制文件描述符。它们的主要区别在于如何指定新的文件描述符以及处理新文件描述符的方式。 dup函数 #i…...
Spring Boot实战 | 如何整合高性能数据库连接池HikariCP
专栏集锦,大佬们可以收藏以备不时之需 Spring Cloud实战专栏:https://blog.csdn.net/superdangbo/category_9270827.html Python 实战专栏:https://blog.csdn.net/superdangbo/category_9271194.html Logback 详解专栏:https:/…...
Spring依赖注入
依赖注入底层原理流程图: https://www.processon.com/view/link/5f899fa5f346fb06e1d8f570 Spring中有两种依赖注入的方式 首先分两种: 手动注入自动注入 手动注入 在XML中定义Bean时,就是手动注入,因为是程序员手动给某个属…...
Linux下Jenkins自动化部署SpringBoot应用
Linux下Jenkins自动化部署SpringBoot应用 1、 Jenkins介绍 官方网址:https://www.jenkins.io/ 2、安装Jenkins 2.1 centos下命令行安装 访问官方,点击文档: 点击 Installing Jenkins: 点击 Linux: 选择 Red Hat/…...
【git 学习】--- ubuntu18.04 搭建本地git服务器
在Ubuntu18.04 上简单创建自己的git服务器~ 环境配置 Ubuntu: 18.04git服务器搭建步骤: ##1.安装git sudo apt-get install git##2.添加用户 sudo adduser test_git //test_git -- git用户名##3. 在Git用户的home目录下创建文件夹,作为裸仓库 sudo…...
JAVA电商平台免费搭建 B2B2C商城系统 多用户商城系统 直播带货 新零售商城 o2o商城 电子商务 拼团商城 分销商城
涉及平台 平台管理、商家端(PC端、手机端)、买家平台(H5/公众号、小程序、APP端(IOS/Android)、微服务平台(业务服务) 2. 核心架构 Spring Cloud、Spring Boot、Mybatis、Redis …...
Android 13 Framework 裁剪
裁剪应用 1. 修改 build/core/product.mk 添加PRODUCT_DEL_PACKAGES变量的声明 新增一行_product_single_value_vars PRODUCT_DEL_PACKAGES # The first API level this product shipped with _product_single_value_vars PRODUCT_SHIPPING_API_LEVEL _product_single_val…...
【Axios封装示例Vue2】
文章目录 为什么要封装axios?如何封装axios在Vue组件中使用封装的axios 为什么要封装axios? 在Vue 2项目中,直接在组件中使用axios可能会导致以下问题: 代码重复:每个组件都需要导入axios并编写相似的请求代码&#…...
k8s-----20、持久化存储--PV/PVC
PV/PVC 1、概念1.1 基本定义1.2 生命周期1.3 PV 卷阶段状态 2、 示例2.1 创建pod和PVC 与PV2.2 绑定PV2.3 强制删除pv,pvc2.4 测试 1、概念 1.1 基本定义 PersistentVolume(PV)是集群中由管理员配置的一段网络存储。 它是集群中的资源,就像…...
python matplotlib 生成矢量图
import matplotlib.pyplot as plt plt.savefig(r"xxx.svg", format"svg")注意: plt.savefig(r"xxx.svg", format"svg") 需要放在 plt.show()前面 原因:如果在 plt.show()调用后, 实际上已经创建了一…...
机器学习中常见的特征工程处理
一、特征工程 特征工程(Feature Engineering)对特征进行进一步分析,并对数据进行处理。 常见的特征工程包括:异常值处理、缺失值处理、数据分桶、特征处理、特征构造、特征筛选及降维等。 1、异常值处理 具体实现 from scipy.s…...
Spring IOC 和 AOP
核心概念 咱们这节就讲完了,在这节中我们讲了两个大概念,一个叫做IOC,一个叫做DI IOC是什么?是用对象的时候不要自己用new而是由外部提供,而spring在进行实现的时候是谁提供,就是IOC容器给你提供。 DI是什…...
echarts插件-liquidFill(水球图)
echarts插件-liquidFill(水球图) 1.下载2.引入:3.使用 1.下载 echarts.js下载:https://cdnjs.com/libraries/echarts echarts-liquidfill.js下载:https://github.com/ecomfe/echarts-liquidfill 2.引入: …...
c++ vscode cmake debug for mac
1. 下载vscode 2. 安装c插件 参考:C programming with Visual Studio Code 3. 安装llvm,可以使用brew安装 4. 配置llvm到系统环境变量中 5. 编写c代码 6. 编写CMakeLists.txt文件(前提安装cmake) cmake_minimum_required(V…...
17 结构型模式-享元模式
1 享元模式介绍 2 享元模式原理 3 享元模式实现 抽象享元类可以是一个接口也可以是一个抽象类,作为所有享元类的公共父类, 主要作用是提高系统的可扩展性. //* 抽象享元类 public abstract class Flyweight {public abstract void operation(String extrinsicState); }具体享…...
创建Secret(手动)
和创建其他类型的 API 对象(Pod、Deployment、StatefulSet、ConfigMap 等)一样,您也可以先在 yaml 文件中定义好 Secret,然后通过 kubectl apply -f 命令创建。此时,您可以通过如下两种方式在 yaml 文件中定义 Secret&…...
基于PHP的线上购物商城,MySQL数据库,PHPstudy,原生PHP,前台用户+后台管理,完美运行,有一万五千字论文。
目录 演示视频 基本介绍 论文截图 功能结构 系统截图 演示视频 基本介绍 基于PHP的线上购物商城,MySQL数据库,PHPstudy,原生PHP,前台用户后台管理,完美运行,有一万五千字论文。 现如今,购物网站是商业…...
Lua 事件触发机制(注册,触发)
日常工作中经常会用到触发机制,这里就提供一个注册触发机制,在代码中在也不用专门去调用各个模块的接口;只需要触发即可,触发后会自动调用接口 直接上代码 local _EventHandle {}; _EventHandle.listenerHandleIndex 0 _EventH…...
c++ 并发与多线程(12)线程安全的单例模式-1
一、什么是线程安全 在拥有共享数据的多条数据并行执行的程序中,线程安全的代码会通过同步机制保证各个线程都可以正常且正确的执行,不会出现数据污染等意外情况。 二、如何保证线程安全 法1、给共享的资源加把锁,保证每个资源变量每时每刻至多被一个线程占用; 法2、让线…...
Python学习笔记--迭代
一、迭代 什么叫做迭代? 比如在 Java 中,我们通过 List 集合的下标来遍历 List 集合中的元素,在 Python 中,给定一个 list 或 tuple,我们可以通过 for 循环来遍历这个 list 或 tuple ,这种遍历就是迭代。…...
挑战杯推荐项目
“人工智能”创意赛 - 智能艺术创作助手:借助大模型技术,开发能根据用户输入的主题、风格等要求,生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用,帮助艺术家和创意爱好者激发创意、提高创作效率。 - 个性化梦境…...
使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
.Net框架,除了EF还有很多很多......
文章目录 1. 引言2. Dapper2.1 概述与设计原理2.2 核心功能与代码示例基本查询多映射查询存储过程调用 2.3 性能优化原理2.4 适用场景 3. NHibernate3.1 概述与架构设计3.2 映射配置示例Fluent映射XML映射 3.3 查询示例HQL查询Criteria APILINQ提供程序 3.4 高级特性3.5 适用场…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...
C++ Visual Studio 2017厂商给的源码没有.sln文件 易兆微芯片下载工具加开机动画下载。
1.先用Visual Studio 2017打开Yichip YC31xx loader.vcxproj,再用Visual Studio 2022打开。再保侟就有.sln文件了。 易兆微芯片下载工具加开机动画下载 ExtraDownloadFile1Info.\logo.bin|0|0|10D2000|0 MFC应用兼容CMD 在BOOL CYichipYC31xxloaderDlg::OnIni…...
Android第十三次面试总结(四大 组件基础)
Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成,用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机: onCreate() 调用时机:Activity 首次创建时调用。…...
视觉slam十四讲实践部分记录——ch2、ch3
ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...
