题目SPOOLing系统的设计与实现

article/2025/9/11 23:54:29

最近刚刚做的一个课程设计,关于SPOOLing的。

一、算法或原理的实现思想
技术原理
SPOOLing技术可将一台物理I/O设备虚拟为多台逻辑I/O设备,同样允许多个用户共享一台物理I/O设备。SPOOLing技术把所有用户进程的输出都送入输出井,然后再由输出进程完成打印工作,而输出井在磁盘上为共享设备。这样,SPOOLing技术就把打印机等独占设备改造成立共享设备。由于SPOOLing技术实现了多个用户进程共同使用打印机这种独占设备的情况,从而实现了把一个设备当成多个设备来使用,即虚拟设备的功能。SPOOLing技术有输入输出井,输入缓冲区和输出缓冲区,输入进程和输出进程,请求打印队列组成输入输出井,用来收集输入设备输入的数据,输入井模拟脱机输入时的磁盘。输出井是收集用户程序的输出数据。输入缓存区由输入设备送来的数据以后再送入输入井,输出缓存区用来暂存从输入井送来的数据,以后再送给输出设备。输入进程模拟脱机输入时的外围控制机,将用户要求的数据从输入设备,通过输入缓冲区送到输入井。当CPU需要数据时,直接从输入井读入主存;输出进程模拟脱机输出时的外围控制机,把用户要求输出的数据,先从主存送到输出井,待输出设备空闲时,再将输出井中的数据,经过输出缓冲区送到输出设备上。由若干张请求打印表所形成的队列,系统为每个请求打印的进程建立一张请求打印表。然后逐个执行,保证所有请求都被完成。
在这里插入图片描述

重要思想
核心思想:是以联机的方式得到脱机的效果
OS的四大特性之一:虚拟性
二、程序

#include <cstdio>
#include <time.h>
#include <iostream>
#include <thread>
#include <mutex>
#include <atomic>
using namespace std;
宏定义,基本量
#define READY_STATUS                0//就绪状态
#define WAIT_BUFFER_STATUS          1//等待状态1,表示输出井满,请求输出的用户进程等待;
#define WAIT_REQUEST_STATUS         2//等待状态2,表示请求输出井空,SP00LING输出进程等待;
#define WAIT_REQBLOCK_STATUS        3//等待状态3,表示请求输出井满,请求输出的用户进程等待;
#define FINISH_STATUS               4//结束态,进程执行完成。
#define RUN_STATUS                  5//运行态,表示正在运行struct PCB {int id;         //进程标识数int status;     //进程状态int count;      //要输出的文件数int tmp_x;      //进程输出时的临时变量
}pcb[3];int FINISH_STATUS_PCB_CNT;//处于结束态的进程个数//请求输出块reqblock
struct Reqblock{int reqname;//请求进程名int length; //本次输出信息长度int addr;   //信息在输出井的首地址
}reqblock[10];int buffer[2][100]; //两个输出井
int EMPTY_BUF_COUNT[2];  //分别表示两个用户进程可使用的输出井的空间int BUF_BEGIN[2];//buffer[i]的第一个满缓冲指针
int BUF_END[2];//buffer[i]的第一个空缓冲指针int REQBLOCK_COUNT;//请求块结构也有一个计数器
int ptr_begin,ptr_end;//表示请求输出块使用情况void user_thread0();//用户进程0
void user_thread1();//用户进程1
void spooling(int pid);//将用户进程输出送入输出井,将输出井数据输出到显示器
int schedule();//调度进程double random() { //生成0~1随机数static double Seed = 233;  //随机种子,在此处可改变初值Seed = 125.0 * (Seed + 1.0);Seed = Seed - 8192.0 * (int)(Seed/8192);return (Seed+0.5)/8192;
}void init() {//对各进程的PCB、输出请求快、输出井初始化for (int i = 0; i < 3; i++) {pcb[i].id = i;pcb[i].status = READY_STATUS;}pcb[2].status = WAIT_REQUEST_STATUS;for (int i = 0; i < 10; i++) {reqblock[i].reqname = -1;reqblock[i].length = 0;reqblock[i].addr = 0;}EMPTY_BUF_COUNT[0] = EMPTY_BUF_COUNT[1] = 100;BUF_BEGIN[0] = BUF_END[0] = 0;BUF_BEGIN[1] = BUF_END[1] = 1;REQBLOCK_COUNT = 10;ptr_begin = ptr_end = 0;for (int i = 0 ;i < 2; i++) {printf("输入进程%d申请Output文件个数:\n",i);scanf("%d",&pcb[i].count);}FINISH_STATUS_PCB_CNT = 0;
}void user_thread0() {pcb[0].tmp_x =  static_cast<int>(random()*10);//cout <<"第0号进程申请输出"<<pcb[0].tmp_x<<"\n";pcb[0].status = RUN_STATUS;spooling(0);
}void user_thread1() {pcb[1].tmp_x = static_cast<int>(random()*10);//cout <<"第1号进程申请输出"<<pcb[1].tmp_x<<"\n";pcb[1].status = RUN_STATUS;spooling(1);
}void spooling(int pid) {if(pid!=-1){while (EMPTY_BUF_COUNT[pid] == 0) {printf("输出井已满,进程%d等待\n",pid);pcb[pid].status = WAIT_BUFFER_STATUS;//转spoolingspooling(-1);}buffer[pid][BUF_END[pid]] = pcb[pid].tmp_x;//修改空缓冲区
//printf("buff[%d][%d]=%d\n",pid,BUF_END[pid],pcb[pid].tmp_x);BUF_END[pid]++;if (BUF_END[pid] == 100)BUF_END[pid] = 0;EMPTY_BUF_COUNT[pid]--;if (pcb[pid].tmp_x == 0) { //一个文件输出结束printf("进程%d完成一个文件\n",pid);while (REQBLOCK_COUNT == 0) {printf("进程%d输出因缺少空闲请求块等待\n",pid);pcb[pid].status = WAIT_REQBLOCK_STATUS;spooling(-1);}//将文件在输出井的位置和长度填入空闲请求块reqblock[ptr_end].reqname = pid;reqblock[ptr_end].addr = BUF_BEGIN[pid];reqblock[ptr_end].length = BUF_END[pid]>BUF_BEGIN[pid] ? BUF_END[pid]-BUF_BEGIN[pid] :100+BUF_END[pid]-BUF_BEGIN[pid];printf("空闲请求块申请成功,位置:%d,长度%d\n",reqblock[ptr_end].addr,reqblock[ptr_end].length);ptr_end++;BUF_BEGIN[pid] = BUF_END[pid];if (ptr_end == 10)ptr_end = 0;REQBLOCK_COUNT--;//空闲请求块数减1if (pcb[2].status == WAIT_REQUEST_STATUS) {//SPOOLING进程是等待状态//唤醒SPOOLING进程pcb[2].status = RUN_STATUS;//  thread(spooling).join();spooling(-1);}pcb[pid].count--;//完成一个文件cout << "进程" << pid << "完成一个文件,还剩" << pcb[pid].count << "个文件\n";if (pcb[pid].count == 0) {pcb[pid].status = FINISH_STATUS;FINISH_STATUS_PCB_CNT++;cout << "进程" << pid << "运行结束\n";//进程i运行结束return;//转进程调度} else {//本文件未结束,继续输出if (pid==0)user_thread0();elseuser_thread1();}} else {//还有其他文件输出if (pid==0)user_thread0();elseuser_thread1();}}else{cout<<"spooling输出\n";int pid = 2;pcb[pid].status = RUN_STATUS;if (REQBLOCK_COUNT == 10) {//请求输出块空cout<<"spooling请求输出块空\n";if (pcb[0].status==FINISH_STATUS && pcb[1].status==FINISH_STATUS) {//两个请求输出的进程结束了pcb[pid].status = FINISH_STATUS;FINISH_STATUS_PCB_CNT++;//SPOOLING进程结束puts("任务结束");return;}else {pcb[pid].status = WAIT_REQUEST_STATUS;puts("无申请块,SPOOLING进程等待");return;//SPOOLING进程等待}}int buf_begin = reqblock[ptr_begin].addr;int buf_end = (buf_begin + reqblock[ptr_begin].length)%100;int userpid = reqblock[ptr_begin].reqname;printf("下面是来自进程%d的输出:\n",userpid);for (int i = buf_begin; i != buf_end; i=(i+1)%100)printf("%d ",buffer[userpid][i]);puts("");puts("spooling输出完成");EMPTY_BUF_COUNT[userpid] += reqblock[ptr_begin].length;//释放输出井空间for (int i = 0; i < 2; i++)if (pcb[i].status == WAIT_BUFFER_STATUS) {pcb[i].status = RUN_STATUS;//唤醒相应进程printf("进程%d又可以执行了\n",userpid);}ptr_begin++;//释放该请求输出块if (ptr_begin == 10)ptr_begin -= 10;REQBLOCK_COUNT++;for (int i = 0; i < 2; i++)if (pcb[i].status == WAIT_REQBLOCK_STATUS) {pcb[i].status = RUN_STATUS;//唤醒相应进程printf("进程%d又可以执行了\n",userpid);}pcb[pid].status = READY_STATUS;}}int schedule() {float rand_num =random();if (rand_num <= 0.45 && pcb[0].status==READY_STATUS) {puts("调度程序:调用用户进程0");user_thread0();return 0;}if (rand_num <= 0.9 && pcb[1].status==READY_STATUS) {puts("调度程序:调用用户进程1");user_thread1();return 1;}if (pcb[2].status == READY_STATUS) {puts("调度程序:调用spooling");spooling(-1);return 2;}return -1;
}
int main() {init();//对各进程的PCB、输出请求快、输出井初始化while (FINISH_STATUS_PCB_CNT < 3) {schedule();}return 0;
}

三、程序的输入和输出
测试数据1:
输入样例:
输出样例:
在这里插入图片描述
五、感想、体会或收获
这一次课程设计度让我学到了在平时课堂不可能学到的东西。我对这一次课程设计的机会非常珍惜。不一定我的课程设计能够完成得有多么完美,但是我很投入的去研究去学习。完成一个任务我非常开心。一开始任务是任务,到后面任务就成了自己的作品了。总结一下有以下体会。
1、网络真的很强大,用在学习上将是一个非常高效的助手。几乎所有的资料都能够在网上找到。包括相关技术原理,以及讲解的视频,真的很好,B站等,让我学习起来没有那么吃力,整个课程设计下来,我浏览的相关网页非常多。当然网上的东西很乱很杂,自己要能够学会筛选。不能决定对或错的,有个很简单的方法就是去尝试
2、同学间的讨论,这是很重要的。老师毕竟比较忙,或者有时候不好意思麻烦老师。对于课程设计最大的讨论伴侣应该是同学了。和自己班上同学讨论让自己受益匪浅。大家都在研究类似的问题,讨论起来,更能够把思路理清楚,相互帮助提高,可以大大提高效率。
3、敢于攻坚,越是难的问题,越是要有挑战的心理。这样就能够达到废寝忘食的境界。当然这也是不提倡熬夜的,毕竟有了精力才能够打持久战。但是做课设一定要有状态,能够在吃饭,睡觉,上厕所都想着要解决的问题,这样你不成功都难。
4、最好在做课设的过程中能够有记录的习惯,这样在写实验报告时能够比较完整的回忆起中间遇到的各种问题。比如当时我遇到我以前从未遇到过的错误,让我都不知道从何下手。在经过大量的资料查阅之后,我对错误有了一定的了解,并且能够用相应的办法来解决。
5.总而言之我认为操作系统是一门实践性很强的课程,这学期虽然在王老师精心教导下,对OS的理论掌握的很好,但是因为没有实训,所以在实践方面还是比较差的,而老师的这一次实训,很好的起到了一定的弥补作用,十分感激老师的用心良苦,在假期我打算去实现所有的课程设计任务,来更加提高对课程的理解深度,让自己变得越来越好。


http://chatgpt.dhexx.cn/article/rw6fy93t.shtml

相关文章

精确度,准确度,精密度关系

1.精确度&#xff0c;准确度&#xff0c;精密度的关系 三者得关系大体可以理解为&#xff0c;准确度精密度 精确度&#xff0c;准确度反应距离真值得偏差&#xff0c;精密度反应测量得稳定性&#xff0c;精确度反应二者之综合。 三者得主次关系&#xff1a;精密度>准确度 …

Mysql的浮点精确度

1.mysql的用于记录小数的类型有三个float ,decimal 和double他们之间的关系 先创建一个表test都用了float ,decimal 和double 插入一条数据查看发现没有发现精度丢失问题 再插入一条数据&#xff0c;发现精度损失&#xff1a; 查看三个类型的范围&#xff1a; 插入小数的位数多…

验证集精确度和损失同时上升

目录 1. 实验结果2. 分析 1. 实验结果 下图中val_acc&#xff0c;val_loss分别表示验证集精确度和损失&#xff1b;train_acc&#xff0c;train_loss分别表示训练集精确度和损失。验证集精确度一直上升&#xff0c;但是损失在第六个epoch后也开始上升&#xff0c;如何解释&…

JavaScript超大或超小数值精确度丢失解决方案

情景一 接口字段&#xff0c;Number类型数据失真&#xff0c;解决方法可直接让服务端把字段类型改成String类型即可。 情景二 某些特殊场景&#xff0c;需要保留小数点后9位(及其以上)&#xff0c;直接调用Number对象自带的toFixed()函数&#xff0c;会出现小数点后数据失真…

关于JavaScript精确度问题

一、js精确度的安全范围是 -2^53 至 2^53 一旦超过这个范围则无法精确表示 1.解决方法 使用第三方包 JSON-Bigint JSONbig.parse() //转换出来的是一个BigNubmer对象 若要使用则用toString()方法 JSONbig.stringify() 2.当axios获取响应数据时自动会将数据JSON.parse()解析为…

【机器学习】准确率、精确度、召回率和 F1 定义

一、说明 数据科学家选择目标变量后 - 例如他们希望预测电子表格中的“列”&#xff0c;并完成了转换数据和构建模型的先决条件&#xff0c;最后步骤之一是评估模型的性能。 二、混淆矩阵的模型 2.1 混淆矩阵 选择性能指标通常取决于要解决的业务问题。假设您的数据集中有 10…

Python计算分类问题的评价指标(准确率、精确度、召回率和F1值,Kappa指标)

机器学习的分类问题常用评论指标有&#xff1a;准确率、精确度、召回率和F1值&#xff0c;还有kappa指标 。 每次调包去找他们的计算代码很麻烦&#xff0c;所以这里一次性定义一个函数&#xff0c;直接计算所有的评价指标。 每次输入预测值和真实值就可以得到上面的指标值&a…

batch_size对精确度和损失的影响

1 问题 在深度学习的学习过程中&#xff0c;模型性能对batchsize虽然没有学习率那么敏感&#xff0c;但是在进一步提升模型性能时&#xff0c;batch_size就会成为一个非常关键的参数。 batch_size对精度和损失的影响研究。 batch_size [,32,64,128&#xff0c;256] 不同batch_…

准度、精度傻傻分不清?

[导读] 做电子产品&#xff0c;常常遇到测量。此时就难免会关注到精度、准度等概念&#xff0c;遇到不少朋友对这两个概念不清楚&#xff0c;今天就来分享一下这两个概念。最近很忙&#xff0c;更的不及时&#xff0c;实在抱歉。也感谢大家不离不弃&#xff01;对于更文分享这件…

机器学习笔记--classification_report精确度/召回率/F1值

classification_report简介 sklearn中的classification_report函数用于显示主要分类指标的文本报告&#xff0e;在报告中显示每个类的精确度&#xff0c;召回率&#xff0c;F1值等信息。 主要参数: y_true&#xff1a;1维数组&#xff0c;或标签指示器数组/稀疏矩阵&#xf…

YOLOv5~目标检测模型精确度

还是yolo5的基础啊~~ 一些关于目标检测模型的评估指标&#xff1a;IOU、TP&FP&FN&TN、mAP等&#xff0c;并列举了目标检测中的mAP计算。 指标评估(重要的一些定义) IOU 也称重叠度表示计算预测回归框和真实回归框的交并比,计算公式如下: TP&FP&FN&…

睿智的目标检测20——利用mAP计算目标检测精确度

睿智的目标检测20——利用mAP计算目标检测精确度 学习前言GITHUB代码下载知识储备1、IOU的概念2、TP TN FP FN的概念3、precision&#xff08;精确度&#xff09;和recall&#xff08;召回率&#xff09;4、概念举例5、单个指标的局限性 什么是AP绘制mAP 学习前言 好多人都想算…

误差与精度

机械专业用于教授学生误差与精度概念的课程叫做《公差与测量》或者叫做《机械精度设计》&#xff0c;而公差或者精度的本质含义就是误差的大小&#xff0c;公差越小&#xff0c;误差越小&#xff0c;精度越高。所以机械专业这门课其本质教授的还是误差理论。 在学校并没有好好学…

层次分析法 AHP

层次分析法&#xff08;Analytic Hierarchy Process&#xff0c;简称 AHP&#xff09;是对一些较为复杂、较为模糊的问题作出决策的简易方法&#xff0c;它特别适用于那些难于完全定量分析的问题。它是美国运筹学家 T. L. Saaty 教授于上世纪 70 年代初期提出的一种简便、灵活而…

权重计算方法一:层次分析法(AHP)

目录 1.层次分析法原理介绍 2.层次分析法建模步骤 3.案例分析 3.1 题目简述 3.2 确定评价指标&#xff0c;建立层次关系 3.3 构造判断矩阵 3.3.1 标度定义 3.3.2 构造判断矩阵 3.4 一致性检验 3.5 层次总排序 4.代码实现 1.层次分析法原理介绍 关于层次分析法的具体原…

【数学建模】层次分析法(AHP)

层次分析法 文章目录 层次分析法用途一、模型介绍一道典型例题1.建立层次结构2.对于同一层次的个元素关于上一层次中的某一准则的重要性进行两两比较&#xff0c;构造两两比较矩阵&#xff08;判断矩阵&#xff09;。2.1 O-C矩阵&#xff08;确定指标的权重&#xff09;2.2 准则…

什么是AHP 层次分析法?

AHP层次分析法(The analytic hierarchy process)是一种解决多目标复杂问题的定性和定量相结合进行计算决策权重的研究方法。该方法将定量分析与定性分析结合起来&#xff0c;用决策者的经验判断各衡量目标之间能否实现的标准之间的相对重要程度&#xff0c;并合理地给出每个决策…

[评价体系] 2、层次分析法AHP原理、例题

目录 1 基本介绍 2 建立步骤 2.1 建立递阶式层次结构模型 2.2 构造比较判断矩阵 2.3 层次单排序及一致性检验 2.4 层次总排序及其一致性检验 2.5 数据加权 3 案例&#xff1a;某学科创新能力评价指标体系 3.1 构建评价指标体系 3.2 构造判断矩阵及一致性检验 3.3 …

数学建模方法——层次分析法(AHP)

目录&#xff1a; 层次分析法简介层次分析法基本原理构造判断矩阵一致性检验一致性检验通过的判断矩阵求权重 4.1 算数平均法求权重 4.2 几何平均法求权重 4.3 特征值法求权重总结 0. 层次分析法简介 层次分析法&#xff08;Analytic Hierarchy Process&#xff0c;简称AHP&…

层次分析法(AHP)算法简介

一、模型介绍 层次分析法&#xff08;AHP&#xff09;是美国运筹学家萨蒂于上世纪70年代初&#xff0c;为美国国防部研究“根据各个工业部门对国家福利的贡献大小而进行电力分配”课题时&#xff0c;应用网络系统理论和多目标综合评价方法&#xff0c;提出的一种层次权重决策分…