【操作系统实验】实验二 进程管理

实验2-1

使用fork创建进程

编写程序,使用系统调用fork()创建如下的进程树,当此程序运行时,在系统中有一个父进程和多个子进程活动,父进程等子进程运行结束后退出。
设置变量X=0,每一个进程在屏幕上显示不同的字符串,父进程的字符串内容要包括自己的学号、姓名、变量X的值;子进程字符串要包括进程PID、子进程序号(子进程1或2或3)、变量X的值,循环显示4次。每次循环X的值加2。记录屏幕上的显示结果,并分析变量X的变化规律。
说明:getpid()获得进程的ID。

代码

#include<unistd.h>
#include<string.h>
#include<stdlib.h>
#include<stdio.h>
int main()
{
	int i=0,X=0;
    pid_t pid1,pid2;
    pid1 = fork();//创建一个新process
    if(pid1 < 0)
    {
            printf("failed");
            exit(1);
    }
    else if(pid1 == 0)
    {
  		for(int i = 0;i<4;i++)
		{
			printf("Child_1 PID = %d, X = %d\n",getpid(),X);
			X =X + 2; 
		}	             
    }
    else
    {
        pid2 = fork();//创建第二个新进程
        if(pid2 < 0)
        {
            printf("failed");
            exit(1);
        }
        else if(pid2 == 0)
        {
			for(int i = 0;i<4;i++)
			{
				printf("Child_2 PID = %d, X = %d\n",getpid(),X);
				X = X + 2;
			}
    	}		
    	else
		{
			for(int i = 0;i<4;i++)
			{
				printf("Parent StuID = 2018308250217, X=%d\n",getpid(),X);
				X = X +2;
			} 
        }	
    }
    getchar();//为了拦住进程,不让进程结束
    return 0;
}

实验2-2

使用exec替换子进程程序

修改任务1编写的程序,将子进程改为独立的程序,父进程创建子进程并进行程序替换,观察程序执行时屏幕出现的现象,并分析原因。

代码

fork.c

#include<string.h>
#include<stdlib.h>
#include<stdio.h>
int main()
{
	int i=0,X=0;
    pid_t pid1,pid2;
    pid1 = fork();//创建一个新process
    if(pid1 < 0)
    {
            printf("failed");
            exit(1);
    }
    else if(pid1 == 0)
    {
  		for(int i = 0;i<4;i++)
		{
			printf("Child_1 PID = %d, X = %d\n",getpid(),X);
			X =X + 2; 
		}	             
    }
    else
    {
        pid2 = fork();//创建第二个新进程
        if(pid2 < 0)
        {
            printf("failed");
            exit(1);
        }
        if(pid2 == 0)
        {
			excel("./a.out","fork_1",NULL);
			exit(1); 
    	}		
    	else
		{
			sleep(2);
			for(int i = 0;i<4;i++)
			{
				printf("Parent StuID = 2018308250217, X=%d\n",getpid(),X);
				X = X +2;
			} 
        }	
    }
    getchar();//为了拦住进程,不让进程结束
    return 0;
}

fork-1.c

#include <stdio.h>
#include <stdlib.h>
#include <unisted.h>
#include <sys/types.h>

int main()
{
	int i = 0,X = 0;
	pid_t pid;
	pid = vfork();
	if(pid > 0)
	{
		for(i = 0; i < 4; i++){
			printf("Parent StuID = 2018308250217, X = %d\n",X);
			X = X + 2;
		}
	}
	else if(pid == 0)
	{
		sleep(2);
		for(i = 0;i < 4;i++)
		{
			printf("Child PID = %d, X = %d\n",getpid(),X);
			X = X + 2; 
		}
	}
	else
	{
		printf("failed");
		exit(1);
	}
	return 0;
}

实验2-3

分析进程的父进程

在系统中,所有的进程组成一个进程树,但在实际操作中,存在中孤儿进程情况,孤儿进程就是父进程已终止,但是子进程没终止,然后就成孤儿。分析孤儿进程的父进程如何变化。
编写程序,要求如下:
父进程创建子进程;
子进程每2秒输出父进程的ID,一直循环;
父进程3秒后,输出自己的ID,结束进程;
分析程序的运行结果,重点说明子进程的父进程如何变化的。

代码

#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>

int main()
{
	int i;
	int pid, ppid;
	printf("begin\n");
	pid = fork();
	if (pid == -1)
	{
		printf("fork error");
		exit(-1);
	}
	if (pid == 0) //child process
	{ 
	    for (i=0; i<5; i++)
		{
			ppid=getppid();
			printf("This is child. PID of the parent is %d\n", ppid);
			sleep(2);
		}
	}
	else
	{
		sleep(3);
		printf("This is parent. the parent process will stop running\n");
	}
	printf("end\n");
	return 0;
}

实验2-4

共享存储区机制进程通信

编程实现生产者和消费者共享存储区功能。生产者随机产生10个整型数据,写入共享存储区;消费者读出数据,并进行平方和平方根运算后输出。使用系统调用shmget()、shmat()、sgmdt()、shmctl()等, 实现程序。

代码

生产者

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <stdio.h>
#define  KEY  75
#define K 1024
main()
{int shmid, *p, i;
char  * shmaddr;
shmid=shmget(KEY,K,0777|IPC_CREAT);
shmaddr=shmat(shmid,0,0);
p=(int *) shmaddr;
for(i=0;i<10;i++)
{  *p = rand()%100;
   sleep(1);
   printf("%d\n",*p);
   p++;
} 
sleep(5);
shmdt(shmaddr);
shmctl(shmid,IPC_RMID,0);
printf("end\n");
}

消费者

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <stdio.h>
#define  KEY  75
#define K 1024
main()
{int mid,*q, *t, i,num;
 char  *maddr;
 mid=shmget(KEY,K,0777);
 maddr=shmat(mid,0,0);
 q=(int *)maddr;
 sleep(6);
 for(i=0;i<10;i++)
 {
  num=(*q)*(*q);
  printf("%d\n",num);
  q++;
  num=0;
}
 shmdt(maddr);
 shmctl(mid,IPC_RMID,0);
}

实验2-5

消息队列实现进程通信

修改上面的程序。父进程从键盘上接受10个数据,对其求和sum1,子进程求这10个数平方和sum2,使用消息队列方式将结果传给父进程,父进程计算sum1+sum2,打印结果。

代码

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#define KEY 100
#define SIZE 10
struct msgform{
	long mtype;
	char mtext[100];
}msg,smsg;
int main()
{
	int num[SIZE];
	int shmid,i;
	char *shmaddr;
	int sum=0;
	int *p;
	int x;
	int msgid;
	//use shm to deliver data
	shmid=shmget(KEY,sizeof(int)*SIZE,0777|IPC_CREAT);
	if(shmid==-1){
		printf("shmget erroe!\n");
		return;
	}
	shmaddr=shmat(shmid,NULL,0);
	p=(int *)shmaddr;
	//input data
	for(i=0;i<SIZE;i++){
		printf("Input number %d:",i+1);
		scanf("%d",p++);
	}
	p=p-5;
	//compute sum of the five numbers
	for(i=0;i<SIZE;i++){
		sum+=*p++;
	}
	printf("SUM:%d\n",sum);
	//use msg queue to deliver result
	msgid=msgget(KEY+1,0777|IPC_CREAT);
	if(msgid==-1){
		printf("msgget error!");
		return;
	}
	if((x=fork())==-1){
		printf("fork error!\n");
		return;
	}
	if(x==0){// child process
		int sum_power2=0;
		p=p-5;
		for(i=0;i<SIZE;i++){
			int temp=*p++;
			sum_power2=sum_power2+temp*temp;
		}
		//send data
		smsg.mtype=1;
		sprintf(smsg.mtext,"%d",sum_power2);
		if(msgsnd(msgid,&smsg,100,0)==-1){
			printf("msgsnd error!\n");
			return;
		}
		return;
	}else{//parent process
		wait();
		//receive data
		msgrcv(msgid,&msg,100,1,0);
		printf("square sum:%s\n",msg.mtext);
	}
	//release resources
	msgctl(msgid,IPC_RMID,0);
	shmdt(shmaddr);
	shmctl(shmid,IPC_RMID,0);
}
  • 8
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1.基本系统进程   Csrss.exe:这是子系统服务器进程,负责控制Windows创建或删除线程以及16位的虚拟DOS环境。   System Idle Process:这个进程是作为单线程运行在每个处理器上,并在系统不处理其它线程的时候分派处理器的时间。   Smss.exe:这是一个会话管理子系统,负责启动用户会话。   Services.exe:系统服务的管理工具。   Lsass.exe:本地的安全授权服务。   Explorer.exe:资源管理器。   Spoolsv.exe:管理缓冲中的打印和传真作业。   Svchost.exe:这个进程要着重说明一下,有不少朋友都有这种错觉:若是在“任务管理器”中看到多个Svchost.exe在运行,就觉得是有病毒了。其实并不一定,系统启动的时候,Svchost.exe将检查注册表中的位置来创建需要加载的服务列表,如果多个Svchost.exe同时运行,则表明当前有多组服务处于活动状态;多个DLL文件正在调用它。   至于其它一些附加进程,大多为系统服务,是可以酌情结束运行的。由于其数量众多,我们在此也不便于一一列举。   在系统资源紧张的情况下,我们可以选择结束一些附加进程,以增加资源,起到优化系统的作用。在排除基本系统及附加进程后,新增的陌生进程就值得被大家怀疑了。 更多内容请看Windows操作系统安装、系统优化大全、系统安全设置专题,或进入讨论组讨论。
(1)创建生产者消费者线程 在Windows2000环境下,创建一个控制台进程,在此进程中创建n个线程来模拟生产者或者消费者。这些线程的信息由本程序定义的“测试用例文件”中予以指定。 该文件的格式和含义如下: 3 1 P 3 2 P 4 3 C 4 1 4 P 2 5 C 3 1 2 4 第一行说明程序中设置几个临界,其余每行分别描述了一个生产者或者消费者线程的信息。每一行的各字段间用Tab键隔开。不管是消费者还是生产者,都有一个对应的线程号,即每一行开始字段那个整数。第个字段用字母P或者C分是生产者还是消费者。第三个字段表示在进入相应线程后,在进行生产和消费动作前的休眠时间,以秒计时;这样做的目的是可以通过调整这一列参数,控制开始进行生产和消费动作的时间。如果是代表生产者,则该行只有三个字段。如果代表消费者,则该行后边还有若干字段,代表要求消费的产品所对应的生产者的线程号。所以务必确认这些对应的线程号存在并且该线程代表一个生产者。 (2)生产和消费的规则 在按照上述要求创建线程进行相应的读写操作时,还需要符合以下要求: ①共享缓冲存在空闲空间时,生产者即可使用共享缓冲。 ②从上边的测试数据文件例子可以看出,某一生产者生产一个产品后,可能不止一个消费者,或者一个消费者多次地请求消费该产品。此时,只有当所有的消费需求都被满足以后,该产品所在的共享缓冲才可以被释放,并作为空闲空间允许新的生产者使用。 ③每个消费者线程的各个消费需求之间存在先后顺序。例如上述测试用例文件包含一行信息“5 C 3 l 2 4”,可知这代表一个消费者线程,该线程请求消费1,2,4号生产者线程生产的产品。而这种消费是有严格顺序的,消费1号线程产品的请求得到满足后才能继续往下请求2号生产者线程的产品。 ④要求在每个线程发出读写操作申请、开始读写操作和结束读写操作时分别显示提示信息。 (3)相关基础知识 本实验所使用的生产者消费者模型具有如下特点: 本实验的多个缓冲不是环形循环的,也不要求按顺序访问。生产者可以把产品放到目前某一个空缓冲中。 消费者只消费指定生产者的产品。 在测试用例文件中指定了所有的生产和消费的需求,只有当共享缓冲数据满足了所有关于它的消费需求后,此共享缓冲才可以作为空闲空间允许新的生产者使用。 本实验在为生产者分配缓冲时各生产者间必须互斥,此后各个生产者的具体生产活动可以并发。而消费者之间只有在对同一产品进行消费时才需要互斥,同时它们在消费过程结束时需要判断该消费对象是否已经消费完毕并清除该产品。 Windows用来实现同步和互斥的实体。在Windows中,常见的同步对象有:信号量(Semaphore)、互斥量(Mutex)、临界段(CriticalSection)等。使用这些对象都分为三个步骤,一是创建或者初始化:接着请求该同步对象,随即进入临界,这一步对应于互斥量的上锁;最后释放该同步对象,这对应于互斥量的解锁。这些同步对象在一个线程中创建,在其他线程中都可以使用,从而实现同步互斥。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值