操作系统实验2课前说明-fork函数

合集下载

《计算机操作系统》实验指导书

《计算机操作系统》实验指导书

《计算机操作系统》实验指导书(适合于计算机科学与技术专业)湖南工业大学计算机与通信学院二O一四年十月前言计算机操作系统是计算机科学与技术专业的主要专业基础课程,其实践性、应用性很强。

实践教学环节是必不可少的一个重要环节。

计算机操作系统的实验目的是加深对理论教学内容的理解和掌握,使学生较系统地掌握操作系统的基本原理,加深对操作系统基本方法的理解,加深对课堂知识的理解,为学生综合运用所学知识,在Linux环境下调用一些常用的函数编写功能较简单的程序来实现操作系统的基本方法、并在实践应用方面打下一定基础。

要求学生在实验指导教师的帮助下自行完成各个操作环节,并能实现且达到举一反三的目的,完成一个实验解决一类问题。

要求学生能够全面、深入理解和熟练掌握所学内容,并能够用其分析、设计和解答类似问题;对此能够较好地理解和掌握,并且能够进行简单分析和判断;能够熟练使用Linux用户界面;掌握操作系统中进程的概念和控制方法;了解进程的并发,进程之间的通信方式,了解虚拟存储管理的基本思想。

同时培养学生进行分析问题、解决问题的能力;培养学生完成实验分析、实验方法、实验操作与测试、实验过程的观察、理解和归纳能力。

为了收到良好的实验效果,编写了这本实验指导书。

在指导书中,每一个实验均按照该课程实验大纲的要求编写,力求紧扣理论知识点、突出设计方法、明确设计思路,通过多种形式完成实验任务,最终引导学生有目的、有方向地完成实验任务,得出实验结果。

任课教师在实验前对实验任务进行一定的分析和讲解,要求学生按照每一个实验的具体要求提前完成准备工作,如:查找资料、设计程序、完成程序、写出预习报告等,做到有准备地上机。

进行实验时,指导教师应检查学生的预习情况,并对调试过程给予积极指导。

实验完毕后,学生应根据实验数据及结果,完成实验报告,由学习委员统一收齐后交指导教师审阅评定。

实验成绩考核:实验成绩占计算机操作系统课程总评成绩的20%。

指导教师每次实验对学生进行出勤考核,对实验效果作记录,并及时批改实验报告,综合评定每一次的实验成绩,在学期终了以平均成绩作为该生的实验成绩。

fork函数超详解及其用法

fork函数超详解及其用法

3. 进程控制上一页第30 章进程下一页3. 进程控制3.1. fork函数#include <sys/types.h>#include <unistd.h>pid_t fork(void);fork调用失败则返回-1,调用成功的返回值见下面的解释。

我们通过一个例子来理解fork是怎样创建新进程的。

例30.3. fork#include <sys/types.h>#include <unistd.h>#include <stdio.h>#include <stdlib.h>int main(void){pid_t pid;char *message;int n;pid = fork();if (pid < 0) {perror("fork failed");exit(1);}if (pid == 0) {message = "This is the child\n"; n = 6;} else {message = "This is the parent\n"; n = 3;}for(; n > 0; n--) {printf(message);sleep(1);}return 0;}$ ./a.outThis is the childThis is the parentThis is the childThis is the parentThis is the childThis is the parentThis is the child$ This is the childThis is the child这个程序的运行过程如下图所示。

图30.4. fork父进程初始化。

父进程调用fork,这是一个系统调用,因此进入内核。

内核根据父进程复制出一个子进程,父进程和子进程的PCB信息相同,用户态代码和数据也相同。

操作系统实验报告(进程的创建)

操作系统实验报告(进程的创建)

wait(0);printf("parent process doesn't change the glob and loc:\n");printf("glob=%d,loc=%d\n",glob,loc);exit(0);}运行结果:2、理解vofork()调用:程序代码:#include<stdio.h>#include<sys/types.h>#include<unistd.h>int glob=3;int main(void){pid_t pid;int loc=3;if((pid=vfork())<0){printf("vfork() error\n");exit(0);}else if(pid==0){glob++;loc--;printf("child process changes the glob and loc\n");exit(0);}elseprintf ("parent process doesn't change the glob and loc\n");printf("glob=%d,val=%d\n",glob,loc);}运行结果:3、给进程指定一个新的运行程序的函数exec().程序代码:printe1.c代码:#include<stdio.h>int main(int argc,char * argv[]){int n;char * * ptr;extern char * * environ;for(n=0;n<argc;n++)printf("argv[%d]:%s\n",n,argv[n]);for(ptr=environ; * ptr!=0;ptr++)printf("%s\n",* ptr);exit(0);}file4.c代码如下:#include<stdio.h>#include<sys/types.h>#include<unistd.h>#include<sys/wait.h>char * env_list[]={"USER=root","PATH=/root/",NULL};int main(){pid_t pid;if((pid=fork())<0){printf("fork error!\n");exit(0);}else if(pid==0){if(execle("/root/print1","print1","arg1","arg2",(char *)0,env_list)<0) printf("execle error!\n");exit(0);}if((waitpid(pid,NULL,0))<0)printf("WAIT ERROR!\n");exit(0);if((pid=fork())<0){printf("fork error!\n");exit(0);}else if(pid==0){if(execlp("print1","print1","arg1",(char *)0)<0)printf("execle error!\n");exit(0);}exit(0);}运行结果:4、进程终止函数exit()。

linux操作系统下fork函数理解

linux操作系统下fork函数理解

linux操作系统下fork函数理解在Linux操作系统中,fork函数是一个非常重要的系统调用,它用于创建一个新的进程。

本文将详细解释fork函数的作用、用法和实现原理,并介绍如何利用fork函数实现进程间通信以及避免一些常见的问题。

一、fork函数的作用和用法在Linux系统中,fork函数用于创建一个新的进程,该进程是调用fork函数的进程的一个副本。

具体而言,fork函数会创建一个新的进程,称为子进程,而调用fork函数的进程被称为父进程。

子进程从fork函数返回的地方开始执行,而父进程则继续执行fork函数之后的代码。

简单来说,fork函数的作用就是将一个进程复制成两个几乎完全相同的进程,但它们具有不同的进程ID(PID)。

fork函数的用法非常简单,只需要在程序中调用fork()即可。

具体代码如下所示:```c#include <stdio.h>#include <sys/types.h>#include <unistd.h>int main() {pid_t pid = fork();if (pid == 0) {// 子进程代码} else if (pid > 0) {// 父进程代码} else {// fork失败的处理代码}return 0;}```在上述代码中,首先使用pid_t类型的变量pid存储fork函数的返回值。

如果pid等于0,则表示当前执行的是子进程的代码;如果pid大于0,则表示当前执行的是父进程的代码;如果pid小于0,则表示fork函数调用失败。

二、fork函数的实现原理在Linux系统中,fork函数的实现是通过复制父进程的内存空间来创建子进程的。

具体来说,fork函数会创建一个新的进程控制块(PCB),并将父进程的PCB全部复制到子进程的PCB中,包括代码段、数据段、堆栈等。

由于子进程是父进程的一个副本,所以它们的代码和数据是完全相同的。

操作系统实验报告二 进程管理 fork()

操作系统实验报告二  进程管理    fork()

闽南师范大学实验报告
实 验 内 容 与 具 体 步 骤
第3页
闽南师范大学实验报告 通过此次实验对进程运行过程有了初步的了解,对用 fork()函数从已存在的进程创建一个新的 进程,会用 getpid()函数取得进程识别码。
实 验 心 得
注:如果填写内容超出表格,自行添加附页。ຫໍສະໝຸດ 第4页闽南师范大学
实验报告
班级
学号 姓名 成绩
同组人
实验日期
课程名称:操作系统
实验题目:进程管理
1.对理论课中学习的进程、程序等的概念作进一步的理解,明确进程和程序的区别; 2. 加深理解进程并发执行的概念,认识多进程的并发执行的实质;
实 验 目 的 与 要 求
PC 兼容机。Window xp 以上操作系统
} else if (pid == 0) { printf("child1 pid is %d\n",getpid()); for(i=0;i<26;i++) {
sleep(1); printf("%c\n",'A'+i); } } else { pid1=fork(); if(pid1==0){ printf("child2 pid is%d\n",getpid()); for(i=0;i<26;i++) { sleep(1); printf("%c\n",'a'+i); } } else if(pid1>0){ printf("parent pid is%d\n",getppid()); for(i=1;i<=26;i++) { sleep(1); printf("%d\n",i); } } } return 0;

期末 操作系统实验课程设计

期末 操作系统实验课程设计

操作系统实验课程设计(二)(参照实验五)学院:计算机科学与工程专业:信息管理工作与信息系统学号:2008142118 姓名:丁建东一、实验题目:设计一个Shell解释器二、实验目的:本设计的主要目的在于学会如何在Unix系统下创建进程和管理进程。

三、实验内容:实现一个简单的shell(命令行解释器),类似于bash, csh等。

要求实现的shell支持以下内部命令:1.cd <目录>更改当前的工作目录到另一个<目录>。

如果<目录>未指定,输出当前工作目录。

如果<目录>不存在,要求有适当的错误信息提示。

改命令应能够改变PWD的环境变量。

2.echo <内容>显示echo后的内容且换行。

3.help简短概要地输出你的shell的使用方法和基本功能。

4.jobs输出shell当前的一系列子进程,要求提供子进程的命名和PID号。

5.quit, exit, bye退出shell。

所有的内部命令应当优于在$PATH中同名的程序。

任何非内部命令必须请求shell创建一个新进程,且该子进程执行指定的程序。

这个新进程必须继承shell的环境变量和指定的命令行参数。

要求实现的shell支持以下内部命令:Batch Processing 如果shell启动带有一个文件名作为参数,打开该文件并执行文件里所有命令。

待所有进程全部结束退出shell。

四、实验思路:1.所用到的系统函数(1)打开目录void cd()API调用:int chdir(dir);getcwd(dir,dir_max);实现:改变当前目录,并判断目录是否存在。

(2)回应void echo()实现:用户输入字符串,以回车结束输入。

char echo_string[echo_len][echo_max];//用户输入命令,以空格符隔开,存为字符串数组按顺序输出用户输入的字符串。

(3)输出当前子进程Void jobs()API调用:shmget(),shmat()实现:开辟一个共享内存区,一旦创建一个子进程,就把该进程的进程ID和名字记字共享区里,在子进程结束的时候消除该记录。

操作系统实验fork()

操作系统实验fork()

并发程序设计【实验目的】:掌握在程序中创建新进程的方法,观察并理解多道程序并发执行的现象。

【实验原理】:fork():建立子进程。

子进程得到父进程地址空间的一个复制。

返回值:成功时,该函数被调用一次,但返回两次,fork()对子进程返回0,对父进程返回子进程标识符(非0值)。

不成功时对父进程返回-1,没有子进程。

【实验内容】:首先分析一下程序运行时其输出结果有哪几种可能性,然后实际调试该程序观察其实际输出情况,比较两者的差异,分析其中的原因。

void main (void){ int x=5;if( fork(()){x+=30;printf (“%d\n”,x);}elseprintf(“%d\n”,x);printf((“%d\n”,x);}【实验要求】:每个同学必须独立完成本实验、提交实验报告、源程序和可执行程序。

实验报告中必须包含预计的实验结果,关键代码的分析,调试记录,实际的实验结果,实验结果分析等内容。

一.源程序1.1程序.#include<stdio.h>#include<sys/types.h>//pid_t类型的定义#include<unistd.h>//函数fork().getpid()定义void main (void){int x=5;if( fork( ) ){x+=30;printf ("%d\n",x);}elseprintf("%d\n",x);printf("%d\n",x);}1.2预测结果:(1)553535(2)353555(3)535535(4)535355(5)355355(6)355535(7)35351.3实际结果:administrator@ubuntu:~/yanhong$ cc 1.cadministrator@ubuntu:~/yanhong$ ./a.out353555administrator@ubuntu:~/yanhong$ cc 1.cadministrator@ubuntu:~/yanhong$ ./a.out5535351.4结果分析:结果表明,子进程先执行还是父进程先执行是不确定的。

操作系统实验二(进程管理)

操作系统实验二(进程管理)

操作系统进程管理实验实验题目:(1)进程的创建编写一段程序,使用系统调用fork( )创建两个子进程。

当此程序运行时,在系统中有一个父进程和两个子进程活动。

让每一个进程在屏幕上显示一个字符:父进程显示字符“a”;子进程分别显示字符“b”和字符“c”。

试观察记录屏幕上的显示结果,并分析原因。

(2)进程的控制修改已编写的程序,将每个进程输出一个字符改为每个进程输出一句话,在观察程序执行时屏幕上出现的现象,并分析原因。

(3)编制一段程序,使其实现进程的软中断通信。

要求:使用系统调用fork( )创建两个子进程,再用系统调用signal( )让父进程捕捉键盘上来的中断信号(即按Del键);当捕捉到中断信号后,父进程调用系统调用kill( )向两个子进程发出信号,子进程捕捉到信号后分别输出下列信息后终止: Child process 1 is killed by parent! Child process 2 is killed by parent! 父进程等待两个子进程终止后,输出如下的信息后终止: Parent process is killed! 在上面的程序中增加语句signal(SIGINT, SIG_IGN)和signal(SIGQUIT, SIG_IGN),观察执行结果,并分析原因。

(4)进程的管道通信编制一段程序,实现进程的管道通信。

使用系统调用pipe( )建立一条管道线;两个进程P1和P2分别向管道各写一句话: Child 1 is sending a message! Child 2 is sending a message! 而父进程则从管道中读出来自于两个子进程的信息,显示在屏幕上。

要求父进程先接收子进程P1发来的消息,然后再接收子进程P2发来的消息。

实验源程序及报告:(1)、进程的创建#include <stdio.h>int main(int argc, char *argv[]){int pid1,pid2; /*fork first child process*/if ( ( pid1=fork() ) < 0 ){printf( "ProcessCreate Failed!");exit(-1);}if ( ( pid1=fork() ) == 0 ){printf( "b\n" );}/*fork second child process*/if ( ( pid2=fork() ) < 0 ){printf( "ProcessCreate Failed!"); exit(-1);}if ( ( pid2=fork() ) == 0 ){printf( "c\n" );}/*parent process*/else{wait(NULL);printf( "a\n" );exit(0);}return 0;}(2)、进程的控制#include <stdio.h>int main(int argc, char *argv[]){ int pid1,pid2;/*fork first child process*/if ( ( pid1=fork() ) < 0 ){printf( "ProcessCreate Failed!");exit(-1);}if ( ( pid1=fork() ) == 0 ){printf( "This is my Unix OS program!\n" ); }/*fork second child process*/if ( ( pid2=fork() ) < 0 ){printf( "ProcessCreate Failed!");exit(-1);}if ( ( pid2=fork() ) == 0 ){printf( "This is the second Child process!\n" ); }/*parent process*/else{wait(NULL);printf( "This is the Parent process\n" );exit(0);}return 0;}(3)编制一段程序,使其实现进程的软中断通信。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

关于fork函数的多进程编程研究
首先我们来看一下多进程的使用,我们简单的使用fork函数来实现。

第一步:我们man fork一下,发现它所依赖的头文件是:sys/types.h 和unistd.h好吧,于是我们开始来使用它。

代码一:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
main()
{pid_t pid; //pid_t 类型实际上就是int型
pid = fork();
if(pid < 0)
printf("erro \n");
else if(pid == 0){
printf("child \n");
}
else{
printf("parnts \n");
}}
这个就是最简单的fork使用了,编译一下,输出:
[xdyang@SEP4020 learning]$ ./fork
child
parnts
好了,大家就要开始问了,为什么会这样?同样一个if的判断语句,为什么会要进去两次?其实很简单,这个我们今天这段代码所要实现的功能:多进程。

当一个进程调用fork函数后,就会创建一个子进程,子进程会拷贝(见附录二)父进程的代码段,但是拥有自己的数据段。

也就是说,其实我们调用了fork函数后,相当于把这个函数代码复制了一遍,也就是产生了类似下面这样的代码:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
main(){
//进程一
pid_t pid;
pid = fork();
if(pid < 0)
printf("erro \n"); else if(pid == 0){ printf("child \n"); }
else{
printf("parnts \n"); }
//进程二
if(pid < 0)
printf("erro \n"); else if(pid == 0){ printf("child \n"); }
else{
printf("parnts \n"); }
}
好了,这样你就可以理解为什么会有两次的if判断了吧,然而,随机而来的问题就是为什么两次if判断的pid的值是不一样的?其实这就是fork的精髓了,就是所谓的一个fork返回两个值,我是这样理解的,在进程一中,它的返回值为0,那么没有问题,它会打印child,在进入进程二之后,它会再次赋值 (见附录一),这时,它就会被赋值为父进程的ID,就会去打印parents了,当然究竟是先被赋值为0还是先被赋值为父进程的ID,这个是由系统的调用算法所决定。

好了,最基本的东西大家都可以理解了,在上面我们提到说两个进程的数据区是不一样,那我们用下面的这个代码来做个试验
代码二:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
main(){
int i = 0;
pid_t pid;
pid = fork();
if(pid < 0)
printf("erro \n");
else if(pid == 0){
i++;
printf("child is %d\n", i);
}
else{
i++;
printf("parnts is %d\n", i);
}
}
这个函数的编译后面的执行结果就是:
[xdyang@SEP4020 learning]$ ./fork
child is 1
parnts is 1
我们发现,尽管在上面子进程中执行了i++的操作,父进程中的i值并没有变化,这就证明了两者的数据区其实是不在的一起的。

好了,理解了上面两点基本上就理解fork的用法,但是在仍然有人遇到了这样一个问题,代码如下:
代码三:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
main(){
int i = 0;
pid_t pid;
printf("fork");//与代码二相比较多的代码
pid = fork();
if(pid < 0)
printf("erro \n");
else if(pid == 0){
i++;
printf("child is %d\n", i);
}
else{
i++;
printf("parnts is %d\n", i);
}
}
我们预计的打印结果应该是:
forkchild is 1
parnts is 1
然而,事实无情地告诉我们,结果不是这样的,执行结果如下:
[xdyang@SEP4020 learning]$ ./fork
forkchild is 1
forkparnts is 1
fork竟然被打印了两次,这如何解释,难道说在调用fork之前的代码也会执行两次,难道说我们刚刚的理解是错误的,其实不然,我们又看下面这段代码:
代码四:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
main(){
int i = 0;
pid_t pid;
printf("fork\n");//注意与代码三的差别
pid = fork();
if(pid < 0)
printf("erro \n");
else if(pid == 0){
i++;
printf("child is %d\n", i);
}
else{
i++;
printf("parnts is %d\n", i);
}
}
这个程序执行后的结果如下:
[xdyang@SEP4020 learning]$ ./fork
fork
child is 1
parnts is 1
这个时候我们就有疑问了,为什么会出现这样的情况呢,其实就是一个\n的差别,这个就需要了解printf的机制了,其实,printf是在接收到\n才刷新串口缓冲区,在一开始,还没有生出子进程的时候,串口缓冲区就已经有数据fork存在了,但是没有被刷新出来。

这个时候子进程出来,于是父子进程就都拥有了这个串口缓冲区的数据:fork。

于是,就出现了两个fork。

大家可以理解了吧~~呵呵,那这个就到此为止了。

附录一:
仔细研究一下,原来系统是这样的,我们可以把fork分为2步,一步是创建一个新的进程,一步为返回。

当我们执行第一步的时候,就会创建一个新的进程,在返回之前我们就拥有了两个进程,比如我们假设子进程先执行,那么此时创建了一个新的进程后,我们就认为下面的执行都是在子进程中了,所以会有一个返回值,那么子进程执行完成时,就会轮到父进程了,而父进程从什么地方开始呢,它就是从
我们上面我们进程创建之后,fork返回之前开始执行,于是又有了一个新的返回值,这样说理解清楚了吧,呵呵~~
附录二:
介绍一个copy-on-write的概念,也就是写时拷贝的概念,我们知道fork在创建子进程的时候需要拷贝父进程的代码和数据,其实在linux中的真实情况并不是说再调用fork就会进行拷贝的,而是一个共享的概念,就是没有改变的地方我们并不会进行一个拷贝,而是共享原来的,但怎样才能造成现实显示的数据段分开的效果呢,我们会在执行的时候,当发生数据写入的时候进行拷贝,也就是说数据有了变化我们将会进行拷贝,这样就可以提高效率和空间利用率了。

相关文档
最新文档