ToB企服应用市场:ToB评测及商务社交产业平台
标题:
【Linux】历程程序更换
[打印本页]
作者:
宝塔山
时间:
2024-6-23 20:25
标题:
【Linux】历程程序更换
思维导图
学习目标
学习历程更换的原理,掌握一些exec*函数的用法。
一、历程的程序更换的原理
用fork创建子历程后,子历程实行的是和父历程雷同的程序(但有可能实行不同的代码分支),若想让子历程实行另一个程序,每每需要调用一种exec函数。
当历程调用一种exec函数时,该历程的用户空间代码和数据完全被新程序更换,并重新程序的启动例程开始实行。这种更换类似于数据修改时的写时拷贝,不会将代码直接覆盖影响到父历程的后续代码。
历程 = 内核数据结构 + 代码 + 数据,代码和数据是要被更换的,而内核数据结构根本不变,没有释放结构,没有创建新的历程。 我们可以通过代码来查验是否创建了子历程?
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
int main()
{
printf("taskexec.....begin\n");
pid_t id = fork();
if(id == 0)
{
printf("child pid : %d\n", getpid());
sleep(2);
execvpe("./pragma", argv, environ);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
if(WIFEXITED(status))
{
printf("child quit success, child exit code: %d\n", WEXITSTATUS(status));
}
else{
printf("child quit failed\n");
}
}
printf("taskexec.....end\n");
return 0;
}
复制代码
站在被更换的历程的角度来看,本质上就是这个程序被加载到内存中去的。怎么将程序加载到内存中??在Linux体系中,exec*函数类似于加载函数。
为什么我们要先将程序加载到内存中呢??因为冯诺依曼体系结构要求,程序先放入内存中去,CPU只会去内存去探求数据和代码。
二、更换函数
这些函数不是体系调用,而是一些封装函数。真正的体系调用函数是execve函数:
2.1 exec*系列函数
加载过程中需要操作体系进行,这种函数底层包括体系调用,因为要将程序加载到内存中。exec*系列函数实行完毕之后,后续的代码不见是正常的,因为代码被更换了,假如不想让后续代码被更换,我们可以利用多历程,让子历程区完成一些代码覆盖。exec*函数的返回值不消关心,只要更换成功,就不会向后走,反之,假如没有更换成功,就一定往后走。
2.2 利用多历程来进行函数更换
利用子历程进行函数更换操作,防止父历程的后续代码被覆盖,改成多历程版。创建子历程,让子历程自己去更换,父历程进行期待操作:可以让子历程实行父历程的一份代码,大概让子历程实行一份新的代码。创建子历程是为了让子历程去完成工作,在子历程中程序更换会发生写时拷贝。
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
pid_t id = fork();
if(id == 0)
{
sleep(2);
execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
2.3 一系列exec*函数(返回值不重要)
2.3.1 execl函数
int execl(const char* path, const char* arg,...);
复制代码
l(list):列表。 列表来记录命令行中实行的命令。
execl函数的参数:第一个参数path:我们实行的程序需要带路径(怎么找到程序,用户要告诉函数), 后面几个参数是可变参数,在命令行中怎么实行,我们就怎么进行传参。
总结:第一个参数的含义是帮我们怎么找到实行的程序,后面几个参数的含义是我们想怎么进行实行程序。
代码:
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
printf("myprocess end.....\n");
return 0;
}
复制代码
这种函数不止能更换一些Linux指令,还能更换我们所写的程序:
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
pid_t id = fork();
if(id == 0)
{
sleep(2);
execl("./test", "test", NULL);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
#include <iostream>
#include <algorithm>
#include <unistd.h>
#include <sys/types.h>
using namespace std;
int main()
{
cout << "C++: pid: %d\n" << getpid() << endl;
cout << "C++: pid: %d\n" << getpid() << endl;
cout << "C++: pid: %d\n" << getpid() << endl;
return 0;
}
复制代码
2.3.2 execv函数
int execv(const char* path, const char* argv[]);
复制代码
v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。
execv函数的参数:由原来的可变参数变成了指针数组。
代码:
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
char* argv[] = {
"ls",
"-a",
"-l",
NULL
};
pid_t id = fork();
if(id == 0)
{
sleep(2);
//execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
execv("/usr/bin/ls", argv);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
2.3.3 execvp函数
v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。p(path):路径,用户可以不传要实行的文件的路径(但是文件名要传递),直接告诉exec*,我要实行谁就可以。p:查找这个程序,体系会自动在情况变量PATH中进行查找。
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
char* argv[] = {
"ls",
"-a",
"-l",
NULL
};
pid_t id = fork();
if(id == 0)
{
sleep(2);
execvp("ls", argv);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
2.3.4 execvpe函数
v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。p(path):路径,用户可以不传要实行的文件的路径(但是文件名要传递),直接告诉exec*,我要实行谁就可以。p:查找这个程序,体系会自动在情况变量PATH中进行查找。e(environment):情况变量。
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
char* argv[] = {
"test",
NULL
};
char* engv[] = {
"haha=1111111",
"hehe=2222222"
};
pid_t id = fork();
if(id == 0)
{
sleep(2);
execvpe("./test", argv, engv);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
#include <iostream>
#include <algorithm>
#include <unistd.h>
#include <sys/types.h>
using namespace std;
int main(int argc, char* argv[], char* engv[])
{
int i = 0;
for(i = 0; argv[i]; i++)
{
cout << argv[i] << endl;
}
for(i = 0; engv[i]; i++)
{
cout << engv[i] << endl;
}
cout << "C++: pid: %d\n" << getpid() << endl;
cout << "C++: pid: %d\n" << getpid() << endl;
cout << "C++: pid: %d\n" << getpid() << endl;
return 0;
}
复制代码
所以,在一个程序中的情况变量和可实行参数是父历程给予的,我们可以通过extern来观察bash历程给予的情况变量和参数部门。
extern char** environ; // 获取父进程的环境变量
复制代码
参数情况变量有三种情况:
用新的情况变量整体更换
用老的情况便令
只增加某一个情况变量:putenv函数
putenv函数:
#include <stdlib.h>
int putenv(char* string);
复制代码
putenv函数添加的情况变量会被添加在当前历程的情况变量表中。
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("myprocess begin.....\n");
char* argv[] = {
"test",
NULL
};
char* engv[] = {
"haha=1111111",
"hehe=2222222"
};
putenv("papa=333333");
pid_t id = fork();
if(id == 0)
{
extern char** environ;
sleep(2);
execvpe("./test", argv, environ);
exit(1);
}
int status = 0;
pid_t rid = waitpid(id, &status, 0);
if(rid > 0)
{
printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
}
printf("myprocess end.....\n");
return 0;
}
复制代码
三、函数解释
这些函数假如调用成功则加载新的程序从启动代码开始实行,不在返回
假如调用失败返回-1
exec函数只有堕落的返回值,而没有成功的返回值
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/)
Powered by Discuz! X3.4