【Linux】历程程序更换

宝塔山  金牌会员 | 2024-6-23 20:25:51 | 来自手机 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 804|帖子 804|积分 2409

思维导图


学习目标

       学习历程更换的原理,掌握一些exec*函数的用法。
一、历程的程序更换的原理

       用fork创建子历程后,子历程实行的是和父历程雷同的程序(但有可能实行不同的代码分支),若想让子历程实行另一个程序,每每需要调用一种exec函数。
       当历程调用一种exec函数时,该历程的用户空间代码和数据完全被新程序更换,并重新程序的启动例程开始实行。这种更换类似于数据修改时的写时拷贝,不会将代码直接覆盖影响到父历程的后续代码。

       历程 = 内核数据结构 + 代码 + 数据,代码和数据是要被更换的,而内核数据结构根本不变,没有释放结构,没有创建新的历程。 我们可以通过代码来查验是否创建了子历程?
  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. #include <sys/wait.h>
  5. #include <sys/types.h>
  6. int main()
  7. {
  8.   printf("taskexec.....begin\n");
  9.   pid_t id = fork();
  10.   if(id == 0)
  11.   {
  12.     printf("child pid : %d\n", getpid());
  13.     sleep(2);
  14.     execvpe("./pragma", argv, environ);
  15.     exit(1);
  16.   }
  17.   int status = 0;
  18.   pid_t rid = waitpid(id, &status, 0);
  19.   if(rid > 0)
  20.   {
  21.     if(WIFEXITED(status))
  22.     {
  23.       printf("child quit success, child exit code: %d\n", WEXITSTATUS(status));
  24.     }
  25.     else{
  26.       printf("child quit failed\n");
  27.     }
  28.   }
  29.   printf("taskexec.....end\n");
  30.   return 0;
  31. }
复制代码
       站在被更换的历程的角度来看,本质上就是这个程序被加载到内存中去的。怎么将程序加载到内存中??在Linux体系中,exec*函数类似于加载函数。
       为什么我们要先将程序加载到内存中呢??因为冯诺依曼体系结构要求,程序先放入内存中去,CPU只会去内存去探求数据和代码。
二、更换函数

这些函数不是体系调用,而是一些封装函数。真正的体系调用函数是execve函数:

2.1 exec*系列函数

       加载过程中需要操作体系进行,这种函数底层包括体系调用,因为要将程序加载到内存中。exec*系列函数实行完毕之后,后续的代码不见是正常的,因为代码被更换了,假如不想让后续代码被更换,我们可以利用多历程,让子历程区完成一些代码覆盖。exec*函数的返回值不消关心,只要更换成功,就不会向后走,反之,假如没有更换成功,就一定往后走。
2.2 利用多历程来进行函数更换

       利用子历程进行函数更换操作,防止父历程的后续代码被覆盖,改成多历程版。创建子历程,让子历程自己去更换,父历程进行期待操作:可以让子历程实行父历程的一份代码,大概让子历程实行一份新的代码。创建子历程是为了让子历程去完成工作,在子历程中程序更换会发生写时拷贝。
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   pid_t id = fork();
  10.   if(id == 0)
  11.   {
  12.     sleep(2);
  13.     execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
  14.     exit(1);
  15.   }
  16.   int status = 0;
  17.   pid_t rid = waitpid(id, &status, 0);
  18.   if(rid > 0)
  19.   {
  20.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  21.   }
  22.   printf("myprocess end.....\n");
  23.   return 0;
  24. }
复制代码

2.3 一系列exec*函数(返回值不重要)


2.3.1 execl函数

  1. int execl(const char* path, const char* arg,...);
复制代码
l(list):列表。 列表来记录命令行中实行的命令。
execl函数的参数:第一个参数path:我们实行的程序需要带路径(怎么找到程序,用户要告诉函数), 后面几个参数是可变参数,在命令行中怎么实行,我们就怎么进行传参。
总结:第一个参数的含义是帮我们怎么找到实行的程序,后面几个参数的含义是我们想怎么进行实行程序。
代码:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
  10.   printf("myprocess end.....\n");
  11.   return 0;
  12. }
复制代码

这种函数不止能更换一些Linux指令,还能更换我们所写的程序:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   pid_t id = fork();
  10.   if(id == 0)
  11.   {
  12.     sleep(2);
  13.     execl("./test", "test", NULL);
  14.     exit(1);
  15.   }
  16.   int status = 0;
  17.   pid_t rid = waitpid(id, &status, 0);
  18.   if(rid > 0)
  19.   {
  20.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  21.   }
  22.   printf("myprocess end.....\n");
  23.   return 0;
  24. }
复制代码
  1. #include <iostream>
  2. #include <algorithm>
  3. #include <unistd.h>
  4. #include <sys/types.h>
  5. using namespace std;
  6. int main()
  7. {
  8.   cout << "C++: pid: %d\n" << getpid() << endl;
  9.   cout << "C++: pid: %d\n" << getpid() << endl;
  10.   cout << "C++: pid: %d\n" << getpid() << endl;
  11.   return 0;
  12. }
复制代码
2.3.2 execv函数 

  1. int execv(const char* path, const char* argv[]);
复制代码
v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。
execv函数的参数:由原来的可变参数变成了指针数组。
代码:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   char* argv[] = {
  10.     "ls",
  11.     "-a",
  12.     "-l",
  13.     NULL
  14.   };
  15.   pid_t id = fork();
  16.   if(id == 0)
  17.   {
  18.     sleep(2);
  19.     //execl("/usr/bin/ls", "ls", "-a", "-l", NULL);
  20.     execv("/usr/bin/ls", argv);
  21.     exit(1);
  22.   }
  23.   int status = 0;
  24.   pid_t rid = waitpid(id, &status, 0);
  25.   if(rid > 0)
  26.   {
  27.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  28.   }
  29.   printf("myprocess end.....\n");
  30.   return 0;
  31. }
复制代码
2.3.3 execvp函数

       v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。p(path):路径,用户可以不传要实行的文件的路径(但是文件名要传递),直接告诉exec*,我要实行谁就可以。p:查找这个程序,体系会自动在情况变量PATH中进行查找。
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   char* argv[] = {
  10.     "ls",
  11.     "-a",
  12.     "-l",
  13.     NULL
  14.   };
  15.   pid_t id = fork();
  16.   if(id == 0)
  17.   {
  18.     sleep(2);
  19.     execvp("ls", argv);
  20.     exit(1);
  21.   }
  22.   int status = 0;
  23.   pid_t rid = waitpid(id, &status, 0);
  24.   if(rid > 0)
  25.   {
  26.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  27.   }
  28.   printf("myprocess end.....\n");
  29.   return 0;
  30. }
复制代码
2.3.4 execvpe函数

       v(vector):指针数组,将命令全部存储在数组中,再将数组传递给execv函数。p(path):路径,用户可以不传要实行的文件的路径(但是文件名要传递),直接告诉exec*,我要实行谁就可以。p:查找这个程序,体系会自动在情况变量PATH中进行查找。e(environment):情况变量。
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   char* argv[] = {
  10.     "test",
  11.     NULL
  12.   };
  13.   char* engv[] = {
  14.     "haha=1111111",
  15.     "hehe=2222222"
  16.   };
  17.   pid_t id = fork();
  18.   if(id == 0)
  19.   {
  20.     sleep(2);
  21.     execvpe("./test", argv, engv);
  22.     exit(1);
  23.   }
  24.   int status = 0;
  25.   pid_t rid = waitpid(id, &status, 0);
  26.   if(rid > 0)
  27.   {
  28.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  29.   }
  30.   printf("myprocess end.....\n");
  31.   return 0;
  32. }
复制代码
  1. #include <iostream>
  2. #include <algorithm>
  3. #include <unistd.h>
  4. #include <sys/types.h>
  5. using namespace std;
  6. int main(int argc, char* argv[], char* engv[])
  7. {
  8.   int i = 0;
  9.   for(i = 0; argv[i]; i++)
  10.   {
  11.     cout << argv[i] << endl;
  12.   }
  13.   for(i = 0; engv[i]; i++)
  14.   {
  15.     cout << engv[i] << endl;
  16.   }
  17.   cout << "C++: pid: %d\n" << getpid() << endl;
  18.   cout << "C++: pid: %d\n" << getpid() << endl;
  19.   cout << "C++: pid: %d\n" << getpid() << endl;
  20.   return 0;
  21. }
复制代码
       所以,在一个程序中的情况变量和可实行参数是父历程给予的,我们可以通过extern来观察bash历程给予的情况变量和参数部门。
  1. extern char** environ; // 获取父进程的环境变量
复制代码
    

参数情况变量有三种情况:


  • 用新的情况变量整体更换
  • 用老的情况便令
  • 只增加某一个情况变量:putenv函数 
putenv函数:

  1. #include <stdlib.h>
  2. int putenv(char* string);
复制代码
       putenv函数添加的情况变量会被添加在当前历程的情况变量表中。 
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <sys/wait.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. int main()
  7. {
  8.   printf("myprocess begin.....\n");
  9.   char* argv[] = {
  10.     "test",
  11.     NULL
  12.   };
  13.   char* engv[] = {
  14.     "haha=1111111",
  15.     "hehe=2222222"
  16.   };
  17.   putenv("papa=333333");
  18.   pid_t id = fork();
  19.   if(id == 0)
  20.   {
  21.     extern char** environ;
  22.     sleep(2);
  23.     execvpe("./test", argv, environ);
  24.     exit(1);
  25.   }
  26.   int status = 0;
  27.   pid_t rid = waitpid(id, &status, 0);
  28.   if(rid > 0)
  29.   {
  30.     printf("father wait success, child exit code: %d\n", WEXITSTATUS(status));
  31.   }
  32.   printf("myprocess end.....\n");
  33.   return 0;
  34. }
复制代码
三、函数解释




  • 这些函数假如调用成功则加载新的程序从启动代码开始实行,不在返回
  • 假如调用失败返回-1
  • exec函数只有堕落的返回值,而没有成功的返回值



免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

宝塔山

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表