当你在一个终端内,运行一个进程的时候,你会发现,如下:
例如图所示,ps -f 进程创建的父进程为bash,并且每次创建进程的pid都不相同,但其父进程都是bash,pid这里都为2690;
注意:不同终端,bash的pid不同;
可见每当打开一个终端,都会有一个bash一直在;
那么可以理解为:
当一个终端创建,都会有一个pid固定的bash创建,
如果要产生一个进程,先将其PCB复制,再将PCB内的pid+1,进程内容进行写时拷贝,在通过exec(简单可理解为替换)将所需创建进程的内容写入
另外:
大多数的linux用的都说bash
可见其为一个系列的函数
man execl看帮助手册
//path:新替换的程序的路径名称
//arg :传给新程序主函数的第一个参数,一般为程序的名字,
/*(后面的...意思为)
arg 后面是剩余参数列表,参数个数可变,必须以空指针作为最后一个参数,
(例如可变参数的函数使最多的:printf)
*/
int execl(const char* path, const char * arg,...);
int execlp(const char* file, const char * arg,...);//在环境变量PATH
指定的路径里面搜索;
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 execve(const char * path, char* const argv[],char* const
envp[]); //系统调用
//前五个是库函数,最后一个是系统调用。所以本质上,上面5个都是通过第六个系统调用实现
的.
exec系列的使用方法的举例如下:
ps的位置:/usr/bin/ps
#include
#include
#include
int main(int argc,char *argv[],char *envp[])
{
printf("main pid=%d\n",getpid());
//execl("/usr/bin/ps","ps","-f",(char *)0);//只是ps也行,这里为替换ps -f 命令
//execlp("ps","ps","-f",(char *)0);
//execle("/usr/bin/ps","ps","-f",(char *)0,envp);
char *myargv[]={"ps","-f",0};
// execv("/usr/bin/ps",myargv);
// execvp("ps",myargv);
execve("/usr/bin/ps",myargv,envp);
//如果前面的替换实现成功,就不会运行后面的代码,表现为不会运行printf
printf("execvp error\n");
exit(0);
}
注意,运行./test 后的pid为2930;
---------运行ps -f 的pid为2932。
所以,替换实现。
#include
#include
#include
int main()
{
printf("mian pid=%d\n",getpid());
execl("/usr/bin/ps","ps","-f",(char *)0);
printf("execl error\n");
exit(0);
}
fork+exec()是Linux上创建新进程的方式;
#include
#include
#include
#include
#include
int main()
{
printf("main pid=%d,ppid=%d\n",getpid(),getppid());
pid_t pid=fork();
assert(pid != -1);
if(pid == 0)
{
printf("child pid = %d,ppid = %d\n",getpid(),getppid());
execl("/usr/bin/ps","ps","-f",NULL);
printf("execl error\n");
exit(0);
}
wait(NULL);
exit(0);
}
//main.c
#include
#include
#include
#include
#include
#include
#include
int main(int argc,char *argv[],char *envp[])
{
printf("main pid=%d\n",getpid());
pid_t pid=fork();
assert(pid!=-1);
if(pid==0)
{
char *myargv[]={"b","hello","abc","123",(char *)0};
execve("./b",myargv,envp);
perror("execve error");
exit(0);
}
wait(NULL);
printf("main over!\n");
exit(0);
}
//b.c
#include
#include
#include
#include
#include
int main(int argc,char *argv[],char *envp[])
{
printf("b pid=%d\n",getpid());
int i=0;
printf("argc=%d\n",argc);
for(;i<argc;i++)
{
printf("argv[%d]=%s\n",i,argv[i]);
}
for(i=0;envp[i]!=NULL;i++)
{
printf("envp[%d]=%s\n",i,envp[i]);
}
exit(0);
}