Linux进程——exec族函数、exec族函数与fork函数的配合
exec族函数解析
作用
我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不创建新进程,所以前后进程的ID并没有改变。
功能
在调用进程内部执行一个可执行文件。可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。
函数族
exec函数族分别是:execl, execlp, execle, execv, execvp, execvpe
函数原型
#include <unistd.h>
extern char **environ;int execl(const char *path, const char *arg, ...);//常用
int execlp(const char *file, const char *arg, ...);//常用
int execle(const char *path, const char *arg,..., char * const envp[]);
int execv(const char *path, char *const argv[]);//常用
int execvp(const char *file, char *const argv[]);//常用
int execvpe(const char *file, char *const argv[],char *const envp[]);
返回值
exec函数族的函数执行成功后不会返回,调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。
参数说明
path | 可执行文件的路径名字 |
arg | 可执行程序所带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束 |
file | 如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件 |
exec族函数参数极难记忆和分辨,函数名中的字符会给我们一些帮助:
l | 使用参数列表 |
p | 使用文件名,并从PATH环境进行寻找可执行文件 |
v | 先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数 |
e | 多了envp[]数组,使用新的环境变量代替调用进程的环境变量 |
将exac函数归为带l、带p、带v三类来说明参数特点
一、带l的一类exac函数(l表示list),包括execl、execlp、execle,要求将新程序的每个命令行参数都说明为 一个单独的参数。这种参数表以空指针结尾。
1.新建一个echoarg并使用execl函数调用
//文件17.c
#include <unistd.h>
#include <stdio.h>int main()
{printf("before execl\n");if(execl("./echoarg","echoarg","hello","word",NULL) == -1){printf("open execl failed\n");//调用execl失败会返回-1,则会执行if里的代码perror("why");}printf("after execl\n");return 0;
}
//文件echoarg.c
#include <stdio.h>int main(int agrc,char *argv[])
{int i;for(i = 0;i < agrc;i++){printf("argv[%d]:%s\n",i,argv[i]);}return 0;
}
先用gcc编译echoarg.c,生成可执行文件echoarg并放在当前路径下。文件echoarg的作用是打印命令行参数。然后再调用execl,用execl 找到并执行echoarg,将当前进程main替换成进程echoarg,就会执行ehcoarg里面的代码,所以”after execl” 没有在终端被打印出来。
路径对应不上的情况:
//文件16.c
#include <unistd.h>
#include <stdio.h>int main()
{printf("before execl\n");if(execl("/bin/echoarg","echoarg","hello","word",NULL) == -1){printf("open execl failed\n");perror("why");//perror函数会打印错误原因,输入内容后结尾会自动加冒号(:)并换行}printf("after execl\n");return 0;
}
可见echoarg存在于当前路径并不存在于bin路径下,所以调用execl函数无法找到echoarg并执行echoarg,调用失败返回值为-1,则会继续执行if函数里的代码。
注:当前文件格式是./+可执行文件名字,其他路径为 /其他路径/可执行文件名字
2.使用execl函数调用ls指令
//文件18.c
#include <unistd.h>
#include <stdio.h>int main()
{printf("before execl\n");if(execl("/bin/ls","ls",NULL) == -1)//不需要参数则找到并使用ls后以NULL结尾{printf("open execl failed\n");perror("why");}printf("after execl\n");return 0;
}
whereis ls//查看ls的路径
可见打印完“before execl”后使用execl函数使用execl函数找到并调用date可执行文件,编译运行代码会直接显示出当前所有文件,实现功能调用。不会执行if中的代码。
3.使用execl函数调用date指令
//文件19.c
#include <unistd.h>
#include <stdio.h>int main()
{printf("we can use execl to know the time\n");//不需要参数则找到并使用ls后以NULL结尾if(execl("/bin/date","date",NULL) == -1){printf("open execl failed\n");perror("why");}printf("after execl\n");return 0;
}
date//查看系统时间
可见一开始直接用date指令可以看到系统时间,在上述代码中通过使用execl函数找到并调用date可执行文件,编译运行代码会直接显示出系统时间,实现功能调用。
二、带p的一类exac函数,包括execlp、execvp、execvpe,如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件。
1.使用execlp函数调用date指令
//文件20.c
#include <unistd.h>
#include <stdio.h>int main()
{printf("we can use execl to know the time\n");if(execlp("date","date",NULL) == -1){printf("open execl failed\n");perror("why");}printf("after execl\n");return 0;
}
上述代码如果调用execl函数则会调用失败,原因是没有输入可执行文件的正确路径。
而在file参数位置直接输入可执行文件名字,调用execlp函数能通过环境变量PATH查找到可执行文件date并执行。
2.环境变量PATH解读
如果使用的可执行文件在PATH里,则使用execlp函数可以不用详细描述绝对路径。
打印环境变量
echo $PATH
修改环境变量
export PATH=$PATH:/home/CLC/Linux1
//通过pwd查看执行文件的路径并将其添加进环境变量PATH路径
修改完毕后结果是:运行/hone/CLC/Linux1下的可执行文件不需要加./便可直接运行
在当前Linux1文件下运行可执行文件不需要加./,使用cd回到目录文件夹也是输出结果一样
三、带v不带l的一类exac函数,包括execv、execvp、execve,应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。
如char *arg[]这种形式,且arg最后一个元素必须是NULL,例如char *arg[] = {“ls”,NULL};
1.execvp
#include <unistd.h>
#include <stdio.h>int main()
{printf("we can use execl to know the time\n");char *agrv[] = {"ls",NULL};//定义一个指针数组if(execvp("ls",agrv) == -1)//将指针数组的地址作为参数{printf("open execl failed\n");perror("why");}printf("after execl\n");return 0;
}
定义一个指针数组并将其地址作为execvp函数后面的参数,同时要以NULL结尾,便可以找到并执行ls可执行文件。同时file参数位置可直接输入可执行文件名字,不需要加路径。
2.execv
#include <unistd.h>
#include <stdio.h>int main()
{printf("before execv\n");char *agrv[] = {"date",NULL};if(execv("/bin/date",agrv) == -1){printf("open execl failed\n");perror("why");}printf("after execvp\n");return 0;
}
跟execvp一样需要定义一个指针数组并将其地址作为参数部分,但需要加上可执行文件路径名字,date在bin下,所以需要加上/bin/才能正确使用execv函数找到并执行date可执行文件。
exec族函数与fork函数的配合
流程图如下:
当父进程检测到输入为1的时候,创建子进程把配置文件的字段值修改掉。
代码如下:
#include<stdio.h>
#include <unistd.h>
int main()
{int a= 0;int fork_r=0;while(1){printf("please input a num\n");scanf("%d",&a);if(a==1)//输入为1,创建子进程{fork_r=fork();if(fork_r==0)//返回值为两次,等于0为子进程{execl("./changeData","changeData",NULL);//找到并执行changeData}}else{printf("no change success\n");}}return 0;
}
上述代码调用execl函数找到并执行的changeData函数如下
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>int main(int argc,char **argv)
{int fdSrc;char *readBuf = NULL;fdSrc = open("config",O_RDWR);int size = lseek(fdSrc,0,SEEK_END);lseek(fdSrc,0,SEEK_SET);readBuf = (char* )malloc(sizeof(char)*size + 8);int n_read = read(fdSrc,readBuf,size);char *p = strstr(readBuf,"leng=");if(p == NULL){printf("not found\n");exit(-1);}p = p+strlen("leng=");*p = '9';lseek(fdSrc,0,SEEK_SET);int n_write = write(fdSrc,readBuf,strlen(readBuf));close(fdSrc);return 0;
}
编译格式是:
gcc changeData.c -o changeData//-o后面的可执行文件才能被execl调用
原配置文件config的值如下:
peed=5
leng=1
SCORE=90
LEVEL=95
只有当输入的值为1时,才会创建子进程并使用execl函数找到并执行changeData可执行文件,从而将原配置文件config中“leng=1”改成“leng=9”。
相关文章:

Linux进程——exec族函数、exec族函数与fork函数的配合
exec族函数解析 作用 我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不创建新进程,所以前后进程的ID并没有改变。 功能 在调用进程内部…...

客户端缓存技术
客户端缓存技术主要有以下几种: 内存缓存:客户端(如浏览器)会将请求到的资源(如HTML页面、图片文件等)存储在内存中,以便在再次访问相同资源时可以快速获取,减少向服务器的请求次数…...

Leetcode -2
Leetcode Leetcode -263.丑数Leetcode -268.丢失的数字 Leetcode -263.丑数 题目:丑数就是只包含质因数 2、3 和 5 的正整数。 给你一个整数 n ,请你判断 n 是否为 丑数 。如果是,返回 true ;否则,返回 false 。 示例…...

使用 DFS 轻松求解数独难题(C++ 的一个简单实现)
起因 都说懒惰是第一生产力,最近在玩数独游戏的时候,总会遇到拆解数独比较复杂的情况,就想着自己写个代码解题,解放双手。所以很快就写了一个简单的代码求解经典数独。拿来跑了几个最高难度的数独发现确实很爽!虽说是…...

【SQL server】 表结构的约束和维护
表结构的约束和维护 修改表结构 (1)添加列 (2)删除列 (3)修改列alter table 表名 add 新列名 数据类型给员工表添加一列邮箱 alter table People add PeopleMail varchar(200)删除列 alter table People drop column PeopleMain修改列 alter table 表名 alter column 列名 数据…...

竞赛 题目:基于大数据的用户画像分析系统 数据分析 开题
文章目录 1 前言2 用户画像分析概述2.1 用户画像构建的相关技术2.2 标签体系2.3 标签优先级 3 实站 - 百货商场用户画像描述与价值分析3.1 数据格式3.2 数据预处理3.3 会员年龄构成3.4 订单占比 消费画像3.5 季度偏好画像3.6 会员用户画像与特征3.6.1 构建会员用户业务特征标签…...

Vue3-ref、reactive函数的watch
Vue3-ref、reactive函数的watch ref函数的watch 原理:监视某个属性的变化。当被监视的属性一旦发生改变时,执行某段代码。watch 属性中的数据需要具有响应式watch 属性可以使用箭头函数watch 属性可以监视一个或者多个响应式数据,并且可以配…...

【智能家居项目】FreeRTOS版本——多任务系统中使用DHT11 | 获取SNTP服务器时间 | 重新设计功能框架
🐱作者:一只大喵咪1201 🐱专栏:《智能家居项目》 🔥格言:你只管努力,剩下的交给时间! 目录 🍓多任务系统中使用DHT11🍅关闭调度器🍅使用中断 &am…...

鸿蒙APP外包开发需要注意的问题
在进行鸿蒙(HarmonyOS)应用开发时,开发者需要注意一些重要的问题,以确保应用的质量、性能和用户体验。以下是一些鸿蒙APP开发中需要特别关注的问题,希望对大家有所帮助。北京木奇移动技术有限公司,专业的软…...

Redis 19 事务
Redis通过MULTI、EXEC、WATCH等命令来实现事务(transaction)功能。事务提供了一种将多个命令请求打包,然后一次性、按顺序地执行多个命令的机制,并且在事务执行期间,服务器不会中断事务而改去执行其他客户端的命令请求…...

Fabric多机部署启动节点与合约部署
这是我搭建的fabric的网络拓扑 3 个 orderer 节点;组织 org1 , org1 下有两个 peer 节点, peer0 和 peer1; 组织 org2 , org2 下有两个 peer 节点, peer0 和 peer1; 以上是我的多机环境的网络拓扑,使用的是docker搭建的。我的网络…...

WordPress主题WoodMart v7.3.2 WooCommerce主题和谐汉化版下载
WordPress主题WoodMart v7.3.2 WooCommerce主题和谐汉化版下载 WoodMart是一款出色的WooCommerce商店主题,它不仅提供强大的电子商务功能,还与流行的Elementor页面编辑器插件完美兼容。 主题文件在WoodMart Theme/woodmart.7.3.2.zip,核心在P…...

Java 高等院校分析与推荐系统
1)项目简介 随着我国高等教育的大众化,高校毕业生就业碰到了前所未有的压力,高校学生就业问题开始进入相关研究者们的视野。在高校学生供给忽然急剧增加的同时,我国高校大学生的就业机制也在发生着深刻的变化,作为就业…...

【JVM】Java虚拟机
本文主要介绍了JVM的内存区域划分,类加载机制以及垃圾回收机制. 其实JVM的初心,就是让java程序员不需要去了解JVM的细节,它把很多工作内部封装好了.但是学习JVM的内部原理有利于我们深入理解学习Java. 1.JVM的内存区域划分 JVM其实是一个java进程 ; 每个java进程,就是一个jvm…...

业务架构、技术架构、项目管理的有机结合
新入职的创业公司一年不行了。 这一年来没有上班,也因为大龄的问题找不到合适的工作。然后考了几个项目管理证书,又思考了一个技术兑现的问题。 技术本身是架构的执行层面,如果上面的公司战略、业务架构变小,缩水,或者…...

拜耳阵列(Bayer Pattern)以及常见彩色滤波矩阵(CFA)
一、拜耳阵列的来源 图像传感器将光线转化成电流,光线越亮,电流的数值就越大;光线越暗,电流的数值就越小。图像传感器只能感受光的强弱,无法感受光的波长。由于光的颜色由波长决定,所以图像传播器无法记录…...

【信息安全】浅谈IDOR越权漏洞的原理、危害和防范:直接对象引用导致的越权行为
前言 ┌──────────────────────────────────┐ │ 正在播放《越权访问》 - Hanser │ ●━━━━━━─────── 00:00 / 03:05 │ ↻ ◁ ❚❚ ▷ ⇆ └───────────────────────────────…...

uni-app 蓝牙打印, CPCL指令集使用
先上代码: GitHub - byc233518/uniapp-bluetooth-printer-demo: 使用uniApp 连接蓝牙打印机 Demo, CPCL 指令简单实用示例 (内含 芝珂,佳博,精臣 多个厂家指令集使用文档) 文件结构: ├── App.vue ├── CPCL 指令手册.pdf // 指令集参考手册 ├── LICENSE ├── R…...

vue-组件通信(二)
🌈个人主页:前端青山 🔥系列专栏:Vue篇 🔖人终将被年少不可得之物困其一生 依旧青山,本期给大家带来vue篇专栏内容:vue-组件通信(二) 目录 组件通信(二) (1) props / $emit 1. 父组件向子组…...

2023年【危险化学品经营单位安全管理人员】考试题及危险化学品经营单位安全管理人员模拟试题
题库来源:安全生产模拟考试一点通公众号小程序 危险化学品经营单位安全管理人员考试题是安全生产模拟考试一点通总题库中生成的一套危险化学品经营单位安全管理人员模拟试题,安全生产模拟考试一点通上危险化学品经营单位安全管理人员作业手机同步练习。…...

Uni-App常用事件
Uni-App是一个跨平台的前端开发框架,支持多个平台的应用开发,包括H5、小程序、App等。在Uni-App中,有许多常用的事件可以用来处理用户交互、页面生命周期等方面的逻辑。以下是一些Uni-App中常用的事件: 点击事件(click…...

【笔记 Pytorch】稀疏矩阵、scipy.sparse模块的使用
安装:pip install scipy 描述:就是专门为了解决稀疏矩阵而生。导入模块:from scipy import sparse 优缺点总结 七种矩阵类型描述coo_matrix ★【名称】coordinate format 【优点】 ① 不同稀疏格式间转换效率高(特别是CSR和CSC) …...

C#学习相关系列之Linq常用方法---排序(一)
一、构建数据 public class Student_1{public int ID { get; set; }public string Name { get; set; }public int Chinese { get; set; }public int Math { get; set; }public int English { get; set; }public override string ToString(){return string.Format("ID:{0},…...

Android Proguard混淆
关于作者:CSDN内容合伙人、技术专家, 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 ,擅长java后端、移动开发、人工智能等,希望大家多多支持。 目录 一、导读二、概览三、语法规则3.1 输入/输出选项3.2 保留选项3.3 缩…...

MySQL 1、初识数据库
一、什么是数据库? 以特定的格式保存好的文件,我们就叫做数据库。 提供较为便捷的数据的存取服务的软件集合、解决方案,我们就叫它数据库。 存储数据用文件就可以了,为什么还要弄个数据库。 文件或数据库都可以存储数据&#…...

H5ke11--3介绍本地,会话存储
代码顺序: 1.设置input,捕获input如果有多个用属性选择符例如 input[typefile]点击事件.向我们的本地存储设置键值对 2.在点击事件外面设置本地存储表示初始化的值.点击上面的事件才能修改我们想修改的值 会话(session)浏览a数据可以写到本地硬盘,关闭页面数据就没了 本地(…...

技术分享 | 如何写好测试用例?
对于软件测试工程师来说,设计测试用例和提交缺陷报告是最基本的职业技能。是非常重要的部分。一个好的测试用例能够指示测试人员如何对软件进行测试。在这篇文章中,我们将介绍测试用例设计常用的几种方法,以及如何编写高效的测试用例。 ## 一…...

quarkus的一些注解1
path 用于指定一个类或者方法的URL路径前缀。 Inject 将一个依赖注入到一个类或方法中 Get 用于指定一个处理HTTP GET请求 Produce 注解用于指定一个方法返回的内容类型。例如,Produces(MediaType.TEXT_PLAIN) 表示该方法返回一个纯文本类型的内容 QuarkusIn…...

初学Redis(Redis的启动以及字符串String)
首先使用在Windows PowerShell中输入指令来启动Redis: redis-server.exe 然后通过指令连接Redis: redis-cli 上图的127.0.0.1是计算机的回送地址 ,6379是默认端口 上述代码中创建了两个键,注意Redis中严格区分大小写࿰…...