进程的概念
在操作系统的角度来看进程是操作系统分配资源的最小单位。
简单来说进程就是处于执行期的程序(目标码存放在某种存储介质上)。但进程并不局限于一段可执行程序代码(代码段)。通常进程还要包含其他资源,像打开的文件(即在Linux中对应的文件描述符),挂起的信号,内核内部数据,处理器的状态,一个或多个具有内存映射的内存地址空间及一个或多个执行线程,当然包括用来存放全局变量的数据段等。
进程的创建
进程的创建使用fork()函数,该函数通过系统调用复制一个现有进程来创建一个全新的进程,调用fork()的进程称为父进程,新产生的进程称为子进程。进程的标识符为一个pid_t类型,实际上就是一个int型。
fork系统调用从内核返回两次:一次返回到父进程,另一次返回到新产生的子进程。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc, char *argv[])
{
pid_t pid;
pid = fork();
if(pid < 0)
{
perror("fork()");
exit(0);
}
if(pid == 0)//子进程
{
printf("[%d]:child process is working!\n",getpid());
}
else //父进程
{
sleep(3);
printf("[%d]:parent process is working!\n",getpid());
}
return 0;
}
父子进程的执行顺序取决于操作系统的调度调度策略。
fork后子进程和父进程的区别:fork后的返回值不一样,PID,PPID,子进程不继承父进程的未决信号和文件锁。
所有进程的进程号都是顺次向下使用的,在所有进程中有一个init进程,也就是通过ps命令可以查看到的1号进程,该进程是所有进程的祖先进程。
进程的消亡
进程的消亡通过wait()函数,父进程通过wait()函数阻塞等待,直至子进程执行完后,等待其进程状态发生变化然后对其进行“收尸”,同样可以使用waitpid()函数实现此功能。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
int main(int argc, char *argv[])
{
pid_t pid;
printf("[%d]:Begin..............\n",getpid());
pid = fork();
if(pid < 0)
{
perror("fork()");
exit(0);
}
if(pid == 0) //子进程
{
printf("[%d]:child process is working!\n",getpid());
}
else //父进程
{
sleep(3); //休眠的原因是让子进程先运行
printf("[%d]:parent process is working!\n",getpid());
wait(NULL); //阻塞等待子进程结束对其进行收尸
}
printf("[%d]:End................\n",getpid());
return 0;
}