实验一进程创建 Linux实验报告

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

实验一进程创建

【实验目的和要求】

1、1.了解进程的概念及意义;

2.了解子进程和父进程;

3.掌握创建进程的方法。

【实验内容】

1、1.子进程和父进程的创建;

2.编写附件中的程序实例;

3.撰写实验报告。

【实验原理】

1、原型:

#include

pid_t fork(void);

在linux中fork函数时非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。fork函数创建子进程的过程为:使用fork函数得到的子进程是父进程的一个复制品,它从父进程继承了进程的地址空间,包括进程上下文、进程堆栈、内存信息、打开的文件描述符、信号控制设定、进程优先级、进程组号、当前工作目录、根目录、资源限制、控制终端,而子进程所独有的只有它的进程号、资源使用和计时器等。通过这种复制方式创建出子进程后,原有进程和子进程都从函数fork返回,各自继续往下运行,但是原进程的fork返回值与子进程的fork返回值不同,在原进程中,fork返回子进程的pid,而在子进程中,fork返回0,如果fork返回负值,表示创建子进程失败。

子进程和父进程继续执行fork调用之后的指令。子进程是父进程的副本。例如,子进程获得父进程数据空间、堆和栈的副本。注意,这是子进程所有用的副本。父进程和子进程并不共享这些存储空间部分。父进程和子进程共享正文段。

2、原型:

#include

pid_t vfork(void);

vfork函数的调用序列和返回值与fork相同,但两者的语义不同。vfork函数用于创建一个新进程,而该进程的目的是exec一个新程序。vfork不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec,它在父进程的空间中运行。vfork保证子进程先运行,在它调用exit之后父进程才可能被调度运行,当子进程调用这两个函数中的任意一个时,父进程会恢复运行。

【程序代码】

1、1_fork.c

#include

#include

#include

int var = 10;

int main(int argc,char *argv[])

{

pid_t pid;

int num = 9;

pid = fork();

if(pid<0)//fork函数创建进程失败!

{

perror("fork");

}

if(pid==0)//子进程

{

var++;

num++;

printf("in son process var = %d,num=%d\n",var,num); }

else//父进程

{

sleep(1);

printf("in father process var=%d,num=%d\n",var,num); }

printf("common code area\n");

return 0;

}

2、2_vfork.c

#include

#include

#include

int var = 10;

int main(int argc,char *argv[])

{

pid_t pid;

int num = 9;

pid = vfork();//创建进程

if(pid<0)

{

perror("vfork");

}

if(pid == 0)

{

var++;

num++;

printf("in son process var=%d,num=%d\n",var,num);

_exit(0);

}

else

{

printf("in father process var=%d,num=%d\n",var,num);

}

return 0;

}

【实验步骤】

一、

1、打开终端,输入命令gedit 1_fork.c,在1_fork.c文件中输入1_fork.bmp 中的代码;

2、输入命令gcc 1_fork.c -o 1_fork,回车后显示无错误;

3、输入命令:./1_fork运行程序。

二、

1、打开终端,输入命令gedit 2_vfork.c,在2_vfork.c文件中输入2_vfork.bmp中的代码;

2、输入命令gcc 2_vfork.c -o 2_vfork,回车后显示无错误:

3、输入命令:./2_vfork运行程序。

【实验结果】

1、

从上面可以看到两次的运行结果不一样。我们知道write函数是不带缓存的。因为在fork之前调用write,所以其数据写到标准输出一次。但是,标准 I/O库是带缓存的。如果标准输出连到终端设备,则它是行缓存的,否则它是全缓存的。当以交互方式运行该程序时,只得到printf输出的行一次,其原因是标准输出缓存由新行符刷新。但是当将标准输出重新定向到一个文件时,却得到printf输出行两次。其原因是,在fork之前调用了printf 一次,当调用fork时,该行数据仍在缓存中,然后在父进程数据空间复制到子进程中时,该缓存数据也被复制到子进程中。于是那时父、子进程各自有了带该行内容的缓存。在exit 之前的第二个printf将其数据添加到现存的缓存中。当每个进程终止时,其缓存中的内容被写到相应文件中。

2、

相关文档
最新文档