用fork函数创建子进程后,子进程往往要调用一种exec函数以执行另一个程序,该子进程被新的程序替换,改变地址空间,进程映像和一些属性,但是pid号不变。
execve():
#include <unistd.h>
int execve(const char *filename, char *const argv[], char *const envp[]);
参数含义:
filename:路径名,表示载入进程空间的新程序路径。
argv[]:命令行参数,argv[0]为命令名。
envp[]:新程序的环境变量。
返回值:成功时不会返回,使用时不用检查返回值,可通过errno检查。
以下函数都是根据execve实现:
int execl(const char *path, const char arg, …/ (char *) NULL */);
int execlp(const char *file, const char arg, …/ (char *) NULL */);
int execle(const char *path, const char arg, …/, (char *) NULL, 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[]);
实验代码:
创建子进程,子进程使用execl调用ls查看当前目录下的文件。
exec.c:
使用execl调用shell命令查看文件信息。
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
void exit_status(pid_t p ,int s);
int main(int argc, const char *argv[])
{
pid_t pid,spid;
int s_son=-1;
pid = fork();
if(pid<0){
perror("fork");
return 1;
}else if(pid == 0){
printf("子进程 %d\n",getpid());
if(-1 == execl("/bin/ls","ls","./",NULL))//ls进程替换子进程
{
perror("execl");
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);//返回0
}else if(pid >0){
printf("父进程 %d\n",getpid());
spid = wait(&s_son);//获取子进程退出状态
exit_status(spid,WEXITSTATUS(s_son));//打印
}
while(1)
{
printf("hello\n");
sleep(2);
}
return 0;
}
/*
* 打印退出状态
*/
void exit_status(pid_t p ,int s)
{
switch (s){
case 0:
printf("子进程--%d--退出状态: EXIT_SUCCESS\n",p);
break;
case 1:
printf("子进程--%d--退出状态: EXIT_FAILURE\n",p);
break;
default:
printf("other\n");
break;
}
}
执行结果:
Linux系统基于这些函数实现了“popen”,“system”等函数,这些函数能够直接调用shell等函数。