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”。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/163113.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C/C++数据结构之堆栈(Stack):理解、实现与运用

当我们讨论堆栈时&#xff0c;我们首先需要了解它的概念和基本原理。堆栈是一种后进先出&#xff08;Last In, First Out&#xff0c;LIFO&#xff09;的数据结构&#xff0c;它的操作主要包括压栈&#xff08;Push&#xff09;和弹栈&#xff08;Pop&#xff09;&#xff0c;以…

SEnet注意力机制(逐行代码注释讲解)

目录 ⒈结构图 ⒉机制流程讲解 ⒊源码&#xff08;pytorch框架实现&#xff09;及逐行解释 ⒋测试结果 ⒈结构图 左边是我自绘的&#xff0c;右下角是官方论文的。 ⒉机制流程讲解 通道注意力机制的思想是&#xff0c;对于输入进来的特征层&#xff0c;我们在每一个通道学…

基于STM32的多组外部中断(EXTI)的优化策略与应用

在某些嵌入式应用中&#xff0c;可能需要同时处理多个外部中断事件。STM32系列微控制器提供了多组外部中断线&#xff08;EXTI Line&#xff09;&#xff0c;可以同时配置和使用多个GPIO引脚作为外部中断触发器。为了有效管理和处理多组外部中断&#xff0c;我们可以采取一些优…

Linux非阻塞等待示例

Linux非阻塞等待实例 非阻塞等待的意义&#xff1a;简单的多进程编程示例代码解释 非阻塞等待的意义&#xff1a; 非阻塞等待在多进程编程中的意义主要体现在提高系统的响应性、实现异步任务执行、动态任务管理和多任务协同工作等方面。它允许父进程在等待子进程退出的同时&…

优化|优化求解器自动调参

原文信息&#xff1a;MindOpt Tuner: Boost the Performance of Numerical Software by Automatic Parameter Tuning 作者&#xff1a;王孟昌 &#xff08;达摩院决策智能实验室MindOpt团队成员&#xff09; 一个算法开发者&#xff0c;可能会幻想进入这样的境界&#xff1a;算…

C++题目练习第二十天__有效的完全平方数

题目链接&#xff1a; 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 题目&#xff1a; 给你一个正整数 num 。如果 num 是一个完全平方数&#xff0c;则返回 true &#xff0c;否则返回 false 。 完全平方数 是一个可以写成某个整数的平方的整数。…

Android 弹出自定义对话框

Android在任意Activity界面弹出一个自定义的对话框&#xff0c;效果如下图所示: 准备一张小图片&#xff0c;右上角的小X图标64*64&#xff0c;close_icon.png&#xff0c;随便找个小图片代替&#xff1b; 第一步&#xff1a;样式添加&#xff0c;注意&#xff1a;默认在value…

2023年中职“网络安全“—Web 渗透测试②

2023年中职“网络安全“—Web 渗透测试② Web 渗透测试任务环境说明&#xff1a;1.访问http://靶机IP/web1/,获取flag值&#xff0c;Flag格式为flag{xxx}&#xff1b;2.访问http://靶机IP/web2/,获取flag值&#xff0c;Flag格式为flag{xxx}&#xff1b;3.访问http://靶机IP/web…

git拉取普通idea Java项目module没有build的问题

在不断完成一个项目的时候&#xff0c;会有不断新加的module&#xff0c;我们用git拉取时会发生没有识别新module的情况。 解决方法是右键项目名称&#xff0c;然后点击Open Module Settings 接下来&#xff0c;点击Module&#xff0c;加号&#xff0c;新建Module的名字就是在g…

深度学习乳腺癌分类 计算机竞赛

文章目录 1 前言2 前言3 数据集3.1 良性样本3.2 病变样本 4 开发环境5 代码实现5.1 实现流程5.2 部分代码实现5.2.1 导入库5.2.2 图像加载5.2.3 标记5.2.4 分组5.2.5 构建模型训练 6 分析指标6.1 精度&#xff0c;召回率和F1度量6.2 混淆矩阵 7 结果和结论8 最后 1 前言 &…

sqli-labs关卡18(基于http头部报错盲注)通关思路

文章目录 前言一、靶场通关需要了解的知识点1、什么是http请求头2、为什么http头部可以进行注入 二、靶场第十八关通关思路1、判断注入点2、爆数据库名3、爆数据库表4、爆数据库列5、爆数据库关键信息 总结 前言 此文章只用于学习和反思巩固sql注入知识&#xff0c;禁止用于做…

若依框架数据源切换为pg库

一 切换数据源 在ruoyi-admin项目里引入pg数据库驱动 <dependency><groupId>org.postgresql</groupId><artifactId>postgresql</artifactId><version>42.2.18</version> </dependency>修改配置文件里的数据源为pg spring:d…

测不准原理

测不准原理 算符的对易关系 commutation relation 测不准原理的矢量推导 Schwarz inequality: 设对易关系&#xff1a; 设一个新态&#xff1a; 投影&#xff1a; 那么有&#xff1a; 代回Schwarz inequality 即可证明&#xff1a;

2023OceanBase年度发布会后,有感

很荣幸收到了OceanBase邀请&#xff0c;于本周四&#xff08;11月16日&#xff09;参加了OceanBase年度发布会并参加了DBA老友会&#xff0c;按照理论应该我昨天&#xff08;星期五&#xff09;就回到成都了&#xff0c;最迟今天白天就该把文章写出来了&#xff0c;奈何媳妇儿买…

zsh和ohmyzsh安装指南+插件推荐

文章目录 1. 安装指南2. 插件配置指南3. 参考信息 1. 安装指南 1. 安装 zsh sudo apt install zsh2. 安装 Oh My Zsh 国内访问GitHub sh -c "$(curl -fsSL https://raw.githubusercontent.com/ohmyzsh/ohmyzsh/master/tools/install.sh)"这将安装 Oh My Zsh 和所…

75基于matlab的模拟退火算法优化TSP(SA-TSP),最优路径动态寻优,输出最优路径值、路径曲线、迭代曲线。

基于matlab的模拟退火算法优化TSP(SA-TSP)&#xff0c;最优路径动态寻优&#xff0c;输出最优路径值、路径曲线、迭代曲线。数据可更换自己的&#xff0c;程序已调通&#xff0c;可直接运行。 75matlab模拟退火算法TSP问题 (xiaohongshu.com)

联想系列台式机Win11系统改Win7系统BIOS设置步骤

联想最新一代的台式机默认操作系统Win11&#xff0c;采用UEFIGPT启动模式&#xff0c;并且开启了安全启动功能&#xff0c;一般用户不能直接将Win11改成Win7&#xff0c;如果需要更改操作系统&#xff0c;是需要再BIOS菜单中关闭安全启动功能的&#xff0c;并且把启动模式设置成…

Linux CentOS7 添加网卡

一台主机中安装多块网卡&#xff0c;有许多优势。可以实现多项功能。 为了学习网卡参数的设置&#xff0c;可以为主机添加多块网卡。与添加磁盘一样&#xff0c;要在VMware中设置。利用图形化方式或命令行查看或设置网卡。本文仅作一初步讨论。有关网络参数的设置不在讨论之列…

【Web】Ctfshow SSRF刷题记录1

核心代码解读 <?php $url$_POST[url]; $chcurl_init($url); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $resultcurl_exec($ch); curl_close($ch); ?> curl_init()&#xff1a;初始curl会话 curl_setopt()&#xff1a;会…

C语言进阶第十课 --------文件的操作

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…