进程的创建

进程的创建

1.详解进程创建的几类函数的说明:forkvforkexecsystem

(1)获取ID

#include <sys/types.h>

#include <unistd.h>

pid_t getpid(void)    获取本进程ID。

pid_t getppid(void)        获取父进程ID

(2)启动进程

(a)pid_tfork(void)

功能:创建子进程

     fork的奇妙之处在于它被调用一次,却返回两次,

它可能有三种不同的返回值:

0:  子进程

子进程ID(大于0):父进程

-1: 出错

实例:

#include<sys/types.h>

#include<unistd.h>

intmain()

   {

        pid_t pid;

        /*此时仅有一个进程*/

        pid=fork();

              /*此时已经有两个进程在同时运行*/

        if(pid<0)

             printf("error in fork!");

        else if(pid==0)

             printf("I am the childprocess, ID is %d\n",getpid());

        else

             printf("I am the parentprocess,ID is %d\n",getpid());

}

说明:

在pid=fork()之前,只有一个进程在执行,但在这条语句执行之后,就变成两个进程在执行了,这两个进程的共享代码段,将要执行的下一条语句都是if(pid==0).

两个进程中,原来就存在的那个进程被称作“父进程”,新出现的那个进程被称作“子进程”,父子进程的区别在于进程标识符(PID)不同.

 

(b)pid_t vfork(void)

功能:创建子进程

表头文件: #include<unistd.h>

定义函数: pid_t vfork(void);

函数说明:

vvfork()会产生一个新的子进程,其子进程会复制父进程的数据与堆栈空间,并继承父进程的用户代码,组代码,环境变量、已打开的文件代码、工作目录和资源限制等。

v子进程不会继承父进程的文件锁定和未处理的信号。

v注意,Linux不保证子进程会比父进程先执行或晚执行,因此编写程序时要留意死锁或竞争条件的发生。

 

(c)exec函数族

exec用被执行的程序替换调用它的程序。

区别:

fork创建一个新的进程,产生一个新的PID

exec启动一个新程序,替换原有的进程,因此进程的PID不会改变

更多详情见博文——exec函数族

 

2.举例对比forkvfork的区别,函数出现位置的不同能否区分代码出现的原因?

区别:

(1)fork:子进程拷贝父进程的数据段

vfork:子进程与父进程共享数据段

(2)fork:父、子进程的执行次序不确定

       vfork:子进程先运行,父进程后运行

 

(a)fork函数

#include<unistd.h>

#include<stdio.h>

intmain(void)

{

       pid_t pid;

       int count=0;

        pid = fork();

       count++;

       printf( count = %d\n", count );

        return0

}

输出:

  count = 1

  count = 1

  

  count++被父进程、子进程一共执行了两次,为什么count的第二次输出为什么不为2?

子进程的数据空间、堆栈空间都会从父进程得到一个拷贝,而不是共享

在子进程中对count进行加1的操作,并没有影响到父进程中的count值,父进程中的count值仍然为0

(b)vfork函数

#include <unistd.h>

#include <stdio.h>

int main(void)

{

     pid_t pid;

     int count=0;

     pid = vfork();

     count++; 

    printf( count = %d\n", count );

    return 0

}

输出:

  count = 1

  count = 2

  

  

  思考如何新创建一个进程然后执行一个程序

进程的等待

1.   僵尸进程的来源?如何避免?

来源:

如果一个进程在其终止的时候,自己就回收所有分配给它的资源,系统就不会产生所谓的僵尸进程了。

父进程调用fork创建子进程后,子进程运行直至其终止,它立即从内存中移除,但进程描述符仍然保留在内存中(进程描述符占有极少的内存空间)。

子进程的状态变成EXIT_ZOMBIE,并且向父进程发送SIGCHLD 信号,父进程此时应该调用 wait() 系统调用来获取子进程的退出状态以及其它的信息。在 wait 调用之后,僵尸进程就完全从内存中移除。

因此一个僵尸存在于其终止到父进程调用wait 等函数这个时间的间隙,一般很快就消失,但如果编程不合理,父进程从不调用 wait 等系统调用来收集僵尸进程,那么这些进程会一直存在内存中。

避免:

一、让僵尸进程的父进程来回收,父进程每隔一段时间来查询子进程是否结束并回收,调用wait()或者waitpid(),通知内核释放僵尸进程。

二、采用信号SIGCHLD通知处理,并在信号处理程序中调用wait函数。

三、让僵尸进程变成孤儿进程,由init回收,就是让父亲先死。

2.   waitwaitpid的用法,两者什么时候是相等的?掌握不同情况下的参数和返回值

wait

函数的作用: 进程的等待,阻塞进程,等待某个子进程退出;

函数的原型:  pid_t wait(int *status);

返回值:成功返回子进程PID,出错-1;

 

waitpid

函数的作用:等待退出,等待信号,或者指定的进程结束

函数的原型:pid_t  waitpid(pid_t pid , int *status, intoptions);

函数的参数:

 pid <-1 :等待进程的Pid绝对值的任何的子进程;

 pid=-1, 任何子进程,---等于wait;

 pid=0,

 pid >0, 等待子进程为pid的子进程退出

 

 options:

  WNOHANG:如果没有子进程退出,马上返回不等待

  WUNTRACED:如果子进程进入暂停执行情况,马上返回,

  

  返回值:如果执行成功返回的是子进程的PID,失败-1;

     如果使用WNOHANG的时候,没有子进程退出,

    

进程的退出

1.   掌握exit_exit的异同点,在内核中有什么区别?举例论证

exit,_exit用于终止进程

区别:

_exit:直接使进程停止,清除其使用的内存,并清除缓冲区中内容

exit与 _exit的区别:在停止进程之前,要检查文件的打开情况,并把文件缓冲区中的内容写回文件才停止进程。

exit():正常结束进程

表头文件: #include<stdlib.h>

定义函数: void exit(int status);

函数说明:

exit()用来正常终结目前进程的执行,并把参数status返回给父进程,而进程所有的缓冲区数据会自动写回并关闭未关闭的文件。

_exit():结束进程执行

表头文件: #include<unistd.h>

定义函数: void _exit(int status);

函数说明:

_exit()用来立刻结束目前进程的执行,并把参数status返回给父进程,并关闭未关闭的文件。

此函数调用后不会返回,并且会传递SIGCHLD信号给父进程,父进程可以由wait函数取得子进程结束状态。

 

  2. 掌握returnexit等退出进程的异同点

returnexit()均可用于函数的返回,但return只是本函数的返回,而exit()则是整个程序的退出。

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实验目的 设计一个有 N个进程并行的进程调度程序。采用最高优先级优先的调度算法进行进程调度的模拟。 实验要求 设计一个有 N个进程并行的进程调度程序。采用最高优先级优先的调度算法进行进程调度的模拟。 实验原理 每个进程用一个进程控制块( PCB)表示。进程控制块可以包含进程名、优先级、到达时间、需要运行时间、已用CPU时间、进程状态等等。 进程的运行时间以时间片为单位进行计算。 每个进程的状态可以是就绪 W(Wait)、运行R(Run)、或完成F(Finish)三种状态之一。进程名、优先级、需要运行时间通过键盘输入。就绪进程获得 CPU后都只能运行一个时间片。用已占用CPU时间加1来表示。运行一个时间片后,进程的已占用 CPU时间已达到所需要的运行时间,则撤消该进程,否则将进程的优先级减1(即降低一级),然后把它插入就绪队列等待CPU。每进行一次调度程序都打印一次运行进程、就绪队列、以及各个进程的 PCB,以便进行检查。 重复以上过程,直到所有进程都完成为止。 实验仪器 PC及其LINUX操作系统 实验步骤 调度算法的流程图如下 : 实验内容 程序部分: #include "stdio.h" #include <stdlib.h> #include <conio.h> #define getpch(type) (type*)malloc(sizeof(type)) #define NULL 0 struct pcb { /* 定义进程控制块PCB */ char name[10]; char state; int super; int ntime; int rtime; struct pcb* link; }*ready=NULL,*p; typedef struct pcb PCB; sort() /* 建立对进程进行优先级排列函数*/ { PCB *first, *second; int insert=0; if((ready==NULL)||((p->super)>(ready->super))) /*优先级最大者,插入队首*/ { p->link=ready; ready=p; } else /* 进程比较优先级,插入适当的位置中*/ { first=ready; second=first->link; while(second!=NULL) { if((p->super)>(second->super)) /*若插入进程比当前进程优先数大,*/ { /*插入到当前进程前面*/ p->link=second; first->link=p; second=NULL; insert=1; } else /* 插入进程优先数最低,则插入到队尾*/ { first=first->link; second=second->link; } } if(insert==0) first->link=p; } } void input() /* 建立进程控制块函数*/ { int i,num; //clrscr(); /*清屏*/ printf("\n 请输入进程号?"); scanf("%d",&num;); for(i=0;i<num;i++) { printf("\n 进程号No.%d:\n",i); p=getpch(PCB); printf("\n 输入进程名:"); scanf("%s",p->name); printf("\n 输入进程优先数:"); scanf("%d",&p->super); printf("\n 输入进程运行时间:"); scanf("%d",&p->ntime); printf("\n"); p->rtime=0;p->state='w'; p->link=NULL; sort(); /* 调用sort函数*/ } } int space() { int l=0; PCB* pr=ready; while(pr!=NULL) { l++; pr=pr->link; } return(l); } void disp(PCB * pr) /*建立进程显示函数,用于显示当前进程*/ { printf("\n qname \t state \t super \t ndtime \t runtime \n"); printf("|%s\t",pr->name); printf("|%c\t",pr->state); printf("|%d\t",pr->super); printf("|%d\t",pr->ntime); printf("|%d\t",pr->rtime); printf("\n"); } void check() /* 建立进程查看函数 */ { PCB* pr; printf("\n **** 当前正在运行的进程是:%s",p->name); /*显示当前运行进程*/ disp(p); pr=ready; printf("\n ****当前就绪队列状态为:\n"); /*显示就绪队列状态*/ while(pr!=NULL) { disp(pr); pr=pr->link; } } void destroy() /*建立进程撤消函数(进程运行结束,撤消进程)*/ { printf("\n 进程 [%s] 已完成.\n",p->name); free(p); } void running() /* 建立进程就绪函数(进程运行时间到,置就绪状态*/ { (p->rtime)++; if(p->rtime==p->ntime) destroy(); /* 调用destroy函数*/ else { (p->super)--; p->state='w'; sort(); /*调用sort函数*/ } } void main() /*主函数*/ { int len,h=0; char ch; input(); len=space(); while((len!=0)&&(ready!=NULL)) { ch=getchar(); h++; printf("\n The execute number:%d \n",h); p=ready; ready=p->link; p->link=NULL; p->state='R'; check(); running(); printf("\n 按任一键继续......"); ch=getchar(); } printf("\n\n 进程已经完成.\n"); ch=getchar(); }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值