system()函数启用新进程占有原进程的文件描述符表的问题
我在A程序中占用了/dev/video0这个独占模式的设备文件,在A中用system函数启用了B程序,B程序的代码中并不包含对/dev/video0的访问,但是我发现B程序也占用了/dev/video0,并且我在A程序中关闭了/dev/video0后,A程序不再占用/dev/video0,但是发现B程序依然占用着/dev/video0这是为什么呢?
当你在A程序中打开/dev/video0并启动B程序时,B程序继承了A程序打开的所有文件描述符,包括/dev/video0的文件描述符。这就导致了B程序也可以访问/dev/video0,即使B程序的代码中并不包含对/dev/video0的访问。
当A程序关闭/dev/video0时,只是让A程序不再使用该设备,但是并没有真正地释放该设备的控制权,因为B程序依然在使用它。因此,当你尝试再次打开/dev/video0时,你会得到“设备忙”的错误提示,因为此时该设备已经被B程序占用,而不是被A程序占用。要想让B程序释放/dev/video0,你需要杀死B程序或让B程序自行关闭/dev/video0。
我的目的是在A中启用B程序,并且在A中可以杀死B,并且B不占用/dev/video0,那么有什么办法呢?
您可以考虑在A程序中使用fork()函数创建一个子进程,然后在子进程中调用B程序。这样,子进程和B程序之间是独立的进程,不会共享A程序的资源,包括/dev/video0。同时,由于子进程和A程序是不同的进程,所以A程序可以通过杀死子进程的方式来结束B程序的执行。
示例代码如下:
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>using namespace std;int main() {pid_t pid = fork();if (pid == -1) {// fork() failedcerr << "Failed to fork()" << endl;return 1;} else if (pid == 0) {// child processexecl("/path/to/B", "B", NULL);cerr << "Failed to execute B" << endl;return 1;} else {// parent process// wait for child process to terminateint status;waitpid(pid, &status, 0);if (WIFEXITED(status)) {cout << "Child process exited with status " << WEXITSTATUS(status) << endl;}}return 0;
}
在子进程中使用execl()函数调用B程序。注意,execl()函数的第一个参数是B程序的路径,第二个参数是B程序的名称,第三个参数是一个空指针。execl()函数调用成功后,子进程会替换成B程序的进程,而且B程序与A程序是独立的进程,不会共享任何资源。
在父进程中使用waitpid()函数等待子进程结束,并获取子进程的退出状态。如果子进程正常结束,可以通过WEXITSTATUS()宏获取子进程的退出码。
另外,如果您需要在A程序中占用/dev/video0这个设备文件,建议您在打开和关闭设备文件时加上文件锁,避免其他进程对设备文件的非法访问。
execl()系列函数来实现相同的目的。execl()系列函数会用指定的可执行程序替换当前进程,所以它不会创建新的进程,也不会复制原来进程的文件描述符表。
execl()函数会用指定的可执行程序来替换当前进程,如果替换成功,那么之后的代码将会在新的程序中执行,如果替换失败,则execl()函数会直接返回,并且不会继续执行后面的代码。
在调用execl()函数时,当前进程的代码段、数据段等内存区域会被指定的可执行程序替换,即当前进程会被新的程序所取代,新程序从main()函数开始执行,原先的程序和它所占用的内存空间都会被释放掉,也就是说,调用execl()函数后,当前进程的程序完全变成了新程序。这种方式常常用于实现进程的程序替换,比如常用的Unix命令行工具,如ls、grep、ps等等,这些工具其实都是通过execl()等函数实现的。
不是同一个进程id号。使用fork()创建新进程后,新进程会复制父进程的地址空间、文件描述符表等资源,但是会拥有一个新的进程id号。然后在新进程中使用execl()函数执行B程序时,新进程的地址空间、文件描述符表等都会被替换为B程序的内容,但是进程id号不会改变。所以A程序和B程序是通过两个不同的进程id号来运行的。
使用 vfork() 函数创建新进程时,新进程会共享父进程的地址空间和部分资源,包括文件描述符、共享内存和信号处理函数等。但是,由于 vfork() 函数不会像 fork() 函数一样完全复制父进程的地址空间,所以需要特别注意以下几点:
新进程和原来的父进程有不同的进程ID。在 vfork() 函数创建的新进程中,子进程的进程ID会被设置为0。一旦子进程开始执行,它会通过调用 exec() 函数来加载新的程序代码并替换自己的地址空间,此时子进程的进程ID将会改变。
父进程和子进程共享地址空间。由于新进程和原来的父进程共享同一地址空间,因此它们可以直接访问相同的变量和函数。这也意味着,如果子进程修改了某个变量的值,父进程中对应的变量值也会发生改变。
子进程会阻塞父进程。在 vfork() 函数创建新进程时,父进程会被阻塞,直到子进程执行完毕或者调用 exec() 函数为止。因此,在使用 vfork() 函数时,需要确保子进程能够尽快执行完毕,否则可能会导致父进程长时间被阻塞。
综上所述,虽然新进程和原来的父进程共享部分资源和文件描述符,但是它们的进程ID是不同的,同时也需要注意避免在子进程中修改父进程的数据,以及确保子进程能够尽快执行完毕,以避免对父进程的阻塞。
相关文章:
system()函数启用新进程占有原进程的文件描述符表的问题
我在A程序中占用了/dev/video0这个独占模式的设备文件,在A中用system函数启用了B程序,B程序的代码中并不包含对/dev/video0的访问,但是我发现B程序也占用了/dev/video0,并且我在A程序中关闭了/dev/video0后,A程序不再占…...

nignx(安装,正反代理,安装tomcat设置反向代理,ip透传)
1安装nginx 安装wget Yum install -y wget 下载(链接从官网找到右键获取) 以下过程root 安装gcc Yum -y install gcc c 安装pcre Yum install -y pcre pcre-devel Openssl Yum install -y openssl openssl-devel 安装zlib Yum install -y zlib zlib-devel 安装make Yum inst…...
sklearn模块常用内容解析笔记
文章目录 回归模型评价指标R2_score预备知识R2_score计算公式r2_score使用方法注意事项参考文献回归模型评价指标R2_score 回归模型的性能的评价指标主要有:RMSE(平方根误差)、MAE(平均绝对误差)、MSE(平均平方误差)、R2_score。但是当量纲不同时,RMSE、MAE、MSE难以衡量模…...

我的 System Verilog 学习记录(2)
引言 从本文开始,就开始系统学习 System Verilog ,不只是语法,还有结合 Questa Sim 的实际编程练习、Debug。 本文简单介绍 System Verilog 语言的用途以及学习的必要性。 前文链接: 我的 System Verilog 学习记录(…...
【调研报告】Monorepo 和 Multirepo 的风格对比及使用示例
带有权重的Monorepo和Multirepo对比 功能/特性MonorepoMultirepoMonorepo权重值Multirepo权重值代码管理管理多个代码库更加复杂管理单个代码库更加简单37依赖管理可以简化依赖管理依赖冲突可能会更加困难73构建和部署构建和部署更加容易构建和部署可能需要更多的配置82团队协…...

Retrofit源码分析
文章目录一、简介二、源码分析2.1Retrofit的本质流程2.2源码分析2.2.1 创建Retrofit实例步骤1步骤2步骤3步骤4步骤5总结2.2.2创建网络请求接口的实例外观模式 & 代理模式1.外观模式2. 代理模式步骤3步骤4总结2.2.3执行网络请求同步请求OkHttpCall.execute()1.发送请求过程2…...

Mybatis-Plus入门系列(20) -兼容多种数据库
有道无术,术尚可求,有术无道,止于术。 文章目录前言方案分析1. 分页2. XML自定义SQL案例演示1. 配置2. 简单分页查询3. 带方言的分页查询参考前言 在我们实际开发软件产品过程中,数据库的类型可能不是确定的,也有客户…...

JetPack板块—Android X解析
Android Jetpack简述 AndroidX 是Android团队用于在Jetpack中开发,测试,打包,发布和版本管理的开源项目。相比于原来的Android Support库,AndroidX 可以称得上是一次重大的升级改进。 和Support库一样,AndroidX与Android 操作系…...
C++学习笔记-数字
当我们使用数字时,通常我们使用原始数据类型,例如 int,short,long,float 和 double 等。数字数据类型,它们的可能值和取值范围在讨论 C 数据类型时已经解释了。 C 定义数字 我们已经在之前笔记的各种实例…...

Nginx——Nginx的基础原理
摘要 Nginx 是俄罗斯人编写的十分轻量级的 HTTP 服务器,是一个高性能的HTTP和反向代理服务器,同时也是一个 IMAP/POP3/SMTP 代理服务器。Nginx 是由俄罗斯人 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的,它已经在该站点运行超过两年半了。…...

服务端开发Java之备战秋招面试篇1
在这个面试造火箭工作拧螺丝的时代背景下,感觉不是很好,不过还好也是拿到了还行的offer,准备去实习了,接下来就是边实习边准备秋招了,这半年把(技术栈八股文面经算法题项目)吃透,希望…...
【C++的OpenCV】第三课-OpenCV图像加载和显示
我们开始学习OpenCV一、OpenCV加载图片和显示图片1.1 imread()函数的介绍1.2 cv::namedWindow()函数的介绍1.4 imshow()函数介绍1.5 Mat容器介绍二、 代码实例(带注释)2.1 代码2.2 执行结果一、OpenCV加载图片和显示图片 本章节中,将会学习到…...

【面试1v1实景模拟】Spring事务 一文到底
老面👴:小伙子,了解Spring的事务吗? 解读🔔:这个必须了解,不了解直接挂~😂😂😂,但面试官肯定不是想听你了解两个字,他是想让你简单的介绍下。 笑小枫🍁:了解,事务在逻辑上是一组操作,要么执行,要不都不执行。主要是针对数据库而言的,比如说 MySQL。为…...

Neuron Selectivity Transfer 原理与代码解析
paper:Like What You Like: Knowledge Distill via Neuron Selectivity Transfercode:https://github.com/megvii-research/mdistiller/blob/master/mdistiller/distillers/NST.py本文的创新点本文探索了一种新型的知识 - 神经元的选择性知识,…...
vue项目关闭子页面,并更新父页面的数据
今天下午是一个非常痛苦的,想要实现一个功能: 父页面打开了一个新的页面(浏览器打开一个新的窗口),并在子页面提交数据之后,父页面的数据要同步更新。 难点:父页面是一个表格列表,…...

第五次作业:修改redis的配置文件使得windows的图形界面客户端可以连接redis服务器
1. 安装 Redis 依赖 Redis 是基于 C语言编写的,因此首先需要安装 Redis 所需要的 gcc 依赖: yum install -y gcc tcl 2、上传安装文件 将下载好的 redis-6.2.7.tar.gz 安装包上传到虚拟机的任意目录(一般推荐上传到 /usr/local/src目录&am…...

【11】FreeRTOS的延时函数
目录1.延时函数-介绍2.相对延时函数-解析2.1函数prvAddCurrentTaskToDelayedList-解析2.3滴答定时器中断服务函数xPortSysTickHandler()-解析2.4函数taskSWITCH_DELAYED_LISTS() -解析3.延时函数-实验4.总结1.延时函数-介绍 函数描述vTaskDelay()相对延时xTaskDelayUntil()绝对…...

Vue页面组成及常用属性
一、Vue页面组成 目前的项目中,Vue页面都是采用组件套娃的形式,由一个一个的组件拼接而成整个页面。一个组件就是一个.vue文件。组件通常由template和script两部分组成: template部分:页面展示的具体元素内容,比如文字…...

j6-IO流泛型集合多线程注解反射Socket
IO流 1 JDK API的使用 2 io简介 输入流用来读取in 输出流用来写出Out 在Java中,根据处理的数据单位不同,分为字节流和字符流 继承结构 java.io包: File 字节流:针对二进制文件 InputStream --FileInputStream --BufferedInputStre…...

创业能否成功?这几个因素很重要!
创业能否成功?这几个因素很重要! 2023-02-22 19:06:53 大家好,我是你们熟悉而又陌生的好朋友梦龙,一个创业期的年轻人 上周末跟朋友一起钓鱼,他跟吐槽现在生意越来越难做。他是我身边可以说是创业很成功的例子&#…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...

循环冗余码校验CRC码 算法步骤+详细实例计算
通信过程:(白话解释) 我们将原始待发送的消息称为 M M M,依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)(意思就是 G ( x ) G(x) G(x) 是已知的)࿰…...

【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...
Java - Mysql数据类型对应
Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...

高危文件识别的常用算法:原理、应用与企业场景
高危文件识别的常用算法:原理、应用与企业场景 高危文件识别旨在检测可能导致安全威胁的文件,如包含恶意代码、敏感数据或欺诈内容的文档,在企业协同办公环境中(如Teams、Google Workspace)尤为重要。结合大模型技术&…...

PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...

BCS 2025|百度副总裁陈洋:智能体在安全领域的应用实践
6月5日,2025全球数字经济大会数字安全主论坛暨北京网络安全大会在国家会议中心隆重开幕。百度副总裁陈洋受邀出席,并作《智能体在安全领域的应用实践》主题演讲,分享了在智能体在安全领域的突破性实践。他指出,百度通过将安全能力…...