1、实验一 linux 进程的创建与控制【实验目的】 1、加深对进程概念的理解,明确进程和程序的区别; 2、进一步认识并发执行的实质; 3、分析进程争用资源的现象,学习解决进程互斥的方法; 【实验环境】编程环境:TC 或者 VC操作系统软件: linux 【准备知识】 一基本概念 1、进程的概念;进程与程序的区别。 2、并发执行的概念。 3、进程互斥的概念。 二系统调用 系统调用是一种进入系统空间的办法。通常,在 OS 的核心中都设置了一组用于实现各种系统功能的子程序,并将它们提供给程序员调用。程序员在需要 OS 提供某种服务的时候,便可以调用一条系统调用命令,去实现希望的功能,这就是系统调用。因
2、此,系统调用就像一个黑箱子一样,对用户屏蔽了操作系统的具体动作而只是控制程序的执行速度等。各个不同的操作系统有各自的系统调用,如 windows API,便是 windows 的系统调用,Linux 的系统调用与之不同的是 Linux 由于内核代码完全公开,所以可以细致的分析出其系统调用的机制。 三相关函数。 1 fork( )函数 fork()函数创建一个新进程。 其调用格式为:int fork(); 其中返回 int 取值意义如下: 正确返回: 等于 0:创建子进程,从子进程返回的 ID 值; 大于 0:从父进程返回的子进程的进程 ID 值。 错误返回: 等于1:创建失败。 2 wait(
3、 )函数 wait()函数常用来控制父进程与子进程的同步。在父进程中调用 wait()函数,则父进程被阻塞,进入等待队列,等待子进程结束。当子进程结束时,会产生一个终止状态字,系统会向父进程发出 SIGCHLD 信号。当接到信号后,父进程提取子进程的终止状态字,从 wait()函数返回继续执行原程序。 其调用格式为: #include #include (pid_t) wait(int *statloc); 正确返回:大于 0:子进程的进程 ID 值; 等于 0:其它。 错误返回:等于1:调用失败。 3 exit( )函数 exit()函数是进程结束最常调用的函数,在 main()函数中调用
4、return,最终也是调用 exit()函数。这些都是进程的正常终止。在正常终止时,exit()函数返回进程结束状态。 其调用格式为: #include void exit(int status); 其中 status 为进程结束状态。 4 kill( )函数 kill()函数用于删除执行中的程序或者任务。 其调用格式为: kill(int PID,int IID); 其中:PID 是要被杀死的进程号,IID 为向将被杀死的进程发送的中断号。关于 Linux 下的 C 语言编程1)编辑器可使用 vi2)编译器使用 gcc格式:gcc option filename例如:gcc -o main
5、main.c主要的 option-o 指定输出文件名(不指定则生成默认文件 a.out)其它的参数见帮助(man gcc) 【实验内容和步骤】1进程的创建编写一段程序,使用系统调用 fork()创建两个子进程。当此程序运行时,在系统中有一个父进程和两个子进程活动。让每个进程在屏幕上显示一个字符,父显示“a“,子进程分别显示“b”和“c”,观察记录显示结果,并分析原因。#includemain()int p1,p2;while(p1=fork()= =-1);if(p1=0)putchar(b);elsewhile(p2=fork()=-1);if(p2= =0)putchar(c);else
6、putchar(a);运行运行结果:rootlocalhost # gcc -o lsj lsj.crootlocalhost # ./lsjbcarootlocalhost # ./lsjbcarootlocalhost # ./lsjbcarootlocalhost # ./lsjbarootlocalhost # c./lsjbcarootlocalhost # ./lsjbcarootlocalhost # ./lsjbcarootlocalhost # ./lsjcarootlocalhost # b分析原因:从进程并发执行来看,各种情况都有可能。上面的三个进程没有同步措施,所以父进
7、程与子进程的输出内容会叠加在一起。输出次序带有随机性。由于函数 printf( )在输出字符串时不会被中断,因此,字符串内部字符顺序输出不变。但由于进程并发执行的调度顺序和父子进程抢占处理机问题,输出字符串的顺序和先后随着执行的不同而发生变化。这与打印单字符的结果相同。2.进程控制( 修改已编写的程序,将每个进程输出一个字符改为每个进程输出一句话,在观察程序执行时屏幕上出现的现象,并分析原因。)编写一段程序,使用系统调用 fork()来创建两个子进程,并由父进程重复显示字符串:“parent:”和自己的标识数,而子进程则重复显示字符串“child:”和自己的标识数。#include#inclu
8、demain()int p1,p2;while(p1=fork()=-1);if(p1=0)printf(“childs pid=%dn“,getppid();elsewhile(p2=fork()=-1);if(p2=0)printf(“childs pid=%dn“,getppid();else printf(“parents pid=%dn“,getppid();结果: rootlocalhost # gcc -o lsj lsj.crootlocalhost # ./lsjchild 1s pid=4852child 2s pid=4852parents pid=3224rootloc
9、alhost # ./lsjchild 1s pid=4903child 2s pid=4903parents pid=3224rootlocalhost # ./lsjchild 1s pid=4918parents pid=3224rootlocalhost # child 2s pid=1./lsjchild 1s pid=4945child 2s pid=4945parents pid=3224rootlocalhost # ./lsjchild 1s pid=5270parents pid=3224rootlocalhost # child 2s pid=5270分析原因:进程标识数
10、随机产生,输出次序也有随机性3. 编写一段程序,使用系统调用 fork()来创建一个子进程。子进程通过系统调用 exec( )更换自己的执行代码,显示新的代码“new program.”后,调用 exit()结束。而父进程则调用 waitpid()等待子进程结束,并在子进程结束后显示子进程的标识符,然后正常结束。代码:main()int p;p=fork();if(p=0)execl(“/root/new”,0);exit(1);elseprintf(“parent=%d”,getpid();waitpid();printf(“child=%d”,getpid();新建 C 程序 new.c:
11、mainprintf(“new program.”);分析原因:先编译 root 下的 new.c,产生 new 应用程序,execl(“/root/new”,0);调用root 目录下的 new 应用程序,执行输出 new program.问题讨论1.系统是怎样创建进程的?答:系统创建的第一个进程是 init 进程。系统中所有的进程都是由当前进程使用系统调用fork()创建的。子进程被创建后继承了父进程的资源。 子进程共享父进程的虚存空间。写时拷贝 (copy on write):子进程在创建后共享父进程的虚存内存空间,只是在两个进程中某一个进程需要向虚拟内存写入数据时才拷贝相应部分的虚拟内
12、存。子进程在创建后执行的是父进程的程序代码。2.可执行文件加载时进行了哪些处理?答:当操作系统装载一个可执行文件的时候,首先操作系统判断该文件是否是一个合法的可执行文件。如果是操作系统将按照段表中的指示为可执行程序分配地址空间。3.当首次调用新创建进程时,其入口在哪里?答:fork 系统调用创建的子进程继承了原进程的 context,也就是说 fork 调用成功后,子进程与父进程并发执行相同的代码。但由于子进程也继承了父进程的程序指针,所以子进程是从 fork()后的语句开始执行(也就是新进程调用的入口)。另外 fork 在子进程和父进程中的返回值是不同的。在父进程中返回子进程的 PID,而在子进程中返回 0。所以可以在程序中检查 PID 的值,使父进程和子进程执行不同的分支。
Copyright © 2018-2021 Wenke99.com All rights reserved
工信部备案号:浙ICP备20026746号-2
公安局备案号:浙公网安备33038302330469号
本站为C2C交文档易平台,即用户上传的文档直接卖给下载用户,本站只是网络服务中间平台,所有原创文档下载所得归上传人所有,若您发现上传作品侵犯了您的权利,请立刻联系网站客服并提供证据,平台将在3个工作日内予以改正。