/ `# {' m4 m) C6 d四、设计思想 4 H" R9 k. }! r4.1 设计思路 [2 k8 y; t8 z 对于单道批处理系统,由于在单道批处理系统中,作业一投入运行,它就占有计算机的一切资源直到作业完成为止,因此调度作业时不必考虑它所需要的资源是否得到满足,它所占用的 CPU时限等因素。 " _# W" e+ [* u' k' u: A 作业调度算法:采用先来先服务(FCFS)调度算法,即按作业提交的先后次序进行调度。总是首先调度在系统中等待时间最长的作业。每个作业由一个作业控制块JCB表示,JCB可以包含如下信息:作业名、提交时间、所需的运行时间、所需的资源、作业状态、链指针等等。% H- ^4 {8 {8 G$ r% a8 Y2 Q L; q1 ~. C
作业的状态可以是等待W(Wait)、运行R(Run)和完成F(Finish)三种状态之一。每个作业的最初状态总是等待W。各个等待的作业按照提交时刻的先后次序排队,总是首先调度等待队列中队首的作业。每个作业完成后要打印该作业的开始运行时刻、完成时刻、周转时间和带权周转时间,这一组作业完成后要计算并打印这组作业的平均周转时间、带权平均周转时间。+ K# V/ Q9 }1 b) O: U
而对于多道批处理系统来说,作业调度(响应比)按一定的算法从磁盘上的“输入井”中选择资源能得到满足的作业装入内存,使作业有机会去占用处理器执行。但是,一个作业能否占用处理器,什么时间能够占用处理器,必须由进程调度来决定。所以,作业调度选中 了一个作业且把它装入内存时,就应为该作业创建一个进程,若有多个作业被装入内存,则内存中同时存在多个进程,这些进程的初始状态为就绪状态,然后,由进程调度(优先数)来选择当前可占用处理器的进程,进程运行中由于某种原因状态发生变化,当它让出处理器时,进程调度就再选另一个作业的进程运行。 因此,作业调度与进程调度相互配合才能实现多道作业的并行执行。0 V9 Z9 X$ M1 g% O8 d B6 I, ^
) m Z& U: a3 E4 n4 e9 m3 b8 O H5 ~4.2 单道批处理系统( u1 v# ^' D0 Q4 q7 k" x8 A
4.2.1 代码结构& v2 U* U& S/ Q' j, T& ~
因为这里的单道批处理系统需要实现多种算法,因此使用了 模板方法设计模式 + 策略模式 来进行开发,下面为各种算法的类签名,算法的具体实现可见代码实现。 : N' F. b$ b: B; l+ l8 }) P I9 H, b
/* 单道批处理系统基类 */ - q" @1 s% y$ K: |9 e$ Pclass SchedulingAlgorithm; ' s$ t% c1 E; S- m0 m3 M6 ^6 W; f( q' W O( m7 m# x+ j
/* First-Come First-Served 先来先服务算法 */" C$ {. c H. Q/ R Z' e& k
class FCFSSchedulingAlgorithm : public SchedulingAlgorithm;/ h2 ]! M% P: l+ b
( T6 }/ H0 \/ C9 @6 F A/* Short Job First 短作业优先算法 */& C& o/ @: D# N4 Z8 W
class SJFSchedulingAlgorithm : public SchedulingAlgorithm; 9 D) Y9 Q) Y5 \; p4 A/ R% w 1 u9 r$ E: y. o/* Highest Response Ratio Next 高响应比优先算法 */* s( D# R8 J* a: b4 x; H0 k
class HRRNSchedulingAlgorithm : public SchedulingAlgorithm; / [0 U. I* _9 g, a0 z0 v7 j1 . S8 Z7 [1 M& d; Z5 ^* A2 . z) g; S" v3 ` [2 c3 $ S5 X4 k( w" Z9 d$ j4# @9 x7 O. f/ K% j8 Y" W( y1 O
5 7 h8 ^; \4 Q/ ^7 P6' B3 A. a, a3 A& s" {* J3 S
7# J% S @( u# U: r
8 1 T+ e+ S5 q/ M95 X0 U/ b% |. }
10 5 F$ k- ?! C# A4 a1 t# m E {114 f* @8 p1 \2 G% Y8 H# o
4.2.2 FCFS + R$ Q7 h) j+ m$ K! T) y 因为这里使用优先级队列来实现,所以我们只需要为每种算法定义不同的比较函数即可,比如对于FCFS算法来说,它的比较算法就是比较该作业提交的时间。 8 q5 }; e' m c3 ^' i( V- S2 d- U# o- U9 V; a9 G! l
// First-Come First-Served+ Q4 J3 H% W2 ^& U
class FCFSSchedulingAlgorithm : public SchedulingAlgorithm: W2 y& [, n! _- W T2 {" S
{ 0 C; e/ {4 w4 a3 Spublic:" z1 d( S9 X( ?2 V8 z' A
struct PriorityCmp' g# _! I/ ?, l* z" i3 R- K$ L
{5 D+ f# g+ x! M) A7 F* x
bool operator()(const JCB j1, const JCB j2) / a; ~ P/ O& @/ M" {, s/ K( } {) y4 O) B2 n" G) X; W, O/ M: Z6 @
return j1.submit_time > j2.submit_time; " _1 d# B! w$ l9 A# S* }; k }+ n4 `' c# |. P/ J8 ^
}; 8 w1 w3 w3 Z4 s) l+ s# a) ]7 m% p' @( h
FCFSSchedulingAlgorithm(){} 0 c$ r' ~6 a; c; z0 e};; E7 d8 W; U5 b7 S5 I- y
1 & \6 m! }3 I! P4 T5 t, ]! G1 o2 E2 $ S/ ?$ E2 u$ o" Y# u7 K1 ^8 F9 r3$ o4 [8 f, F! y1 Z; A/ \8 `
4 $ }1 R5 x$ o% u% k% Z# l5: {" O% |+ d6 k+ [# N$ b
64 u( g) h3 W0 l$ ^0 ?
7; M' | G' T7 x! c/ K0 y% @( b
8 0 F+ b3 s; T7 U `3 Y, v' q7 U; I9; v* \: s* |+ F( s' v
10 * Y+ L% C2 \: w' O( _11, `' w" z. d: X8 v/ q
12 : ^# b" o; C' y; q! t" @% o2 {13 : K5 w6 x3 _8 }: f14) ^' j7 u, P" }7 a0 R
4.2.3 SJF5 K7 J5 v& ^5 ]& S
对于SJF算法来说,它的比较函数就是比较作业的时间长度(时间越短优先级越高)。% }8 F+ n3 }- }
7 E( J% A) z7 s4 ?# F
// Short Job First 0 [, l2 a s* B; E0 v: fclass SJFSchedulingAlgorithm : public SchedulingAlgorithm $ u4 T7 R! S% Q0 w+ P, z6 c! ~{$ G$ ?4 N7 F2 h& w0 q
public:+ o: M" x. A- b( J( {. e B/ @
struct PriorityCmp. Z" |0 h" x. v$ K5 T+ K! {# V
{ ( N; r3 J: h' H7 r/ V' F bool operator()(const JCB j1, const JCB j2) 6 ?0 d, Q: {5 [6 P8 A* H { 5 r. u3 `* H, f \. p: n' f; T$ { return j1.required_running_time > j2.required_running_time; - d% m) c% y* m3 G } ) `6 h7 a, H# H6 C/ i P }; - e4 q* E5 h% d* _/ ? ' o3 F0 P A4 e3 k# R* ? SJFSchedulingAlgorithm(){}+ ]3 j) L$ N- f: _7 U
};/ \& g; ^2 T X1 l& ~
1' ~: i1 A: y3 j9 U
2 ) _2 j3 h7 ]7 p4 P7 q33 o. V; i4 l. v3 K5 h+ p& [
4* d V/ O& D1 w" H: J2 M1 C$ E
59 Q' f. d0 Q6 `9 y7 J3 x
6 $ ]$ l* q. [! D, D9 o$ j3 C6 C' R76 y A2 @% `& t$ x% ]
8- K9 y3 t* s1 F- F& r7 b
9 ; ^ |/ ?) H1 Y/ Q) l8 A. E10 m L+ A& J5 t/ J1 j5 T113 @2 T) c5 N4 z1 A
121 H" R% G5 v7 O# \6 w
13 3 W. P6 |: n% L" e% ]3 l. ]14 ' c8 D; z, q5 U/ T; T; T. `4.2.4 HRRN 2 R- z# {( p/ x8 | 该算法的比较函数比较的是每个作业的响应比:(等待时间+要求服务时间)/要求服务时间,响应比越高,优先级越高,越优先被调度。 2 `! _& `1 l9 k! U7 p# z ! g: ~( x" y- M& b) q' l. ?// Highest Response Ratio Next& j. \$ w( {: X' ^, I5 q+ [2 u1 X- b
class HRRNSchedulingAlgorithm : public SchedulingAlgorithm , ]* ]5 A. y' U7 u0 R{ % f, Z" D: n; D# |public: % Z4 n% K+ `6 M: R struct PriorityCmp/ `, y# }1 e& u, ^( }
{ % p6 z8 w; S1 F' V' I k | double getPriority(JCB jcb) 5 V! z% |9 Q: W { " I: s: {6 ^ c2 w+ `$ H return ((cur_time - jcb.submit_time) + jcb.required_running_time) / jcb.required_running_time; # Q2 K' B, `& Q1 \+ C" p }- w, z" V) F$ s' k3 N: w. k
% X! g) k3 e' _8 Y! \6 t) _ bool operator()(const JCB j1, const JCB j2) g! a- ]8 U2 ]8 q% [* f$ n. W! W
{ , ?* e, Q1 D8 d X return getPriority(j1) > getPriority(j2); * E1 R! ]7 y1 m9 h `! B2 c }) Z- `4 { v. [$ U; C) ]+ K
};( Y$ [- E3 t3 \! r$ v0 w
8 e' U2 L" l$ p% I7 @9 n! G/ C HRRNSchedulingAlgorithm(){}; r5 Q% ~/ M6 P
};7 Z9 x5 B! [3 ]+ Q \$ w Z
1- `) N8 U5 j$ J# C) m' d
2; {. p) `) v, e* z- d
35 O1 B5 i; g' _+ Y- X
4# M+ T$ U% }- _9 i* b8 B
5 - ?+ o; i0 U. ] Y6 " C' j- U. R1 H# l5 V" q7 : ~% ?! d/ I/ j4 b7 J8# c; b- |9 B: [
9 $ s: i5 p% H1 A6 u8 b$ ^# M104 W4 N9 |6 e4 L0 A
116 Y5 |- X( E( y7 R; V8 o6 R
12! Z% U: B; ~8 W! Q2 _( m# ]
13" L- K' |# Q6 B0 A
14 ! {; J! H4 U3 ^* R15; N2 t. c+ a# l: f+ O9 a
16 N' G1 L0 l2 i
173 m$ h6 [4 L4 q* R
18, M" j9 V: L" U/ S. Y ?% J% l
19& s+ ^5 v$ x, P( v
4.3 多道批处理系统6 s' F9 m( L1 e8 G$ Y
4.3.1 FCFS + PSA 0 |* D2 k+ F" s/ B8 K$ { 对于多道批处理系统来说,情况会复杂一些,因为对于多道批处理系统,除了作业的调度还存在进程的调度,也就是作业调度决定该任务能不能使用处理机(有没有资格),但就算该任务被调度了,因为多道批处理系统,还需要考虑相关的资源,因此能不能使用到处理机,还要看进程的调度。 2 o7 |% k+ A% J/ g 因此在这里对于作业调度采用了FCFS算法,而对于进程的调度采用了PSA算法(静态优先级,可抢占),并且没有继续使用实验一中的进程调度算法的代码,而是重新写了一遍。 ; y3 _. b7 U d. h- j( G$ N: B 所有的任务初始会被存放在 back_queue 中,而就绪的任务和正在运行的任务都存储于 psa_ready_queue 中,等到 psa_ready_queue 中存在空间时,会通过任务调度算法从 back_queue 中选择合适的任务进行调度,当任务(准确说是进程)被调到 psa_ready_queue 中后,会根据它的优先级判断它是否能够优先运行,如果不能就只能保持就绪状态,一直等到优先级高的进程运行完毕后再运行。 % |6 v9 t. A# G& A7 J 因为这部分代码的逻辑比较复杂,所以就不在这里单独罗列了,具体可见下面的代码实现。需要注意的是多道批处理系统代码逻辑中被注释掉的代码为测试代码,可以使用其来对代码的准确性进行测试。 4 s9 x& j7 Y8 K0 T" K( Z% r/ Y% K; e% u6 U6 a( V
五、代码实现 0 L. i% p; S+ n0 C% s8 s% F, n#include <iostream>, D7 i k3 @4 M6 ?- I+ f
#include <string> 2 N( F, g# Z6 q$ J#include <vector> 5 h% ~$ g' \" `7 Y$ N Q+ N% |8 y6 x#include <queue> ) ]0 t' S7 ]# `#include <cstdlib> 4 d2 q8 ?. Z1 c# O e! Q#include <ctime>. ^% d% ` D0 W- ?5 u3 B
#include <unistd.h>. l: A! w- ?: h3 r8 ~+ Z
#include <iomanip>2 F8 [8 s* ?9 Q; `$ x
( t& m+ N! |) Y# y: g9 Ctypedef int TimeSlice;2 z B z2 U' c. _( d7 f1 H5 i7 y
typedef int Resource; ) C/ _9 G0 J4 Utypedef std::queue<TimeSlice> * JobTimeSeries;3 K& P0 Q$ b* \& ]0 v
: `2 E$ @) P% H; P* g# A( c. s4 e
TimeSlice cur_time = 0; ; V) w* F+ j& ~5 t8 Mint average_turnaround_time = 0; 7 L; [ N2 I s, |3 a. k) _1 Ndouble average_power_turnaround_time = 0;! H b2 `. d0 A1 K _( u8 n
$ P- ~) b7 O" P7 P0 h0 u7 m2 Senum ProcessStatus: T7 f- D. U! K$ i$ T) q
{, C" j' {1 q2 s$ P6 b
Ready = 0,+ s V2 n7 J) d6 S8 p( o. c a& |
Running = 1,) j5 b: f5 `6 x, u
//Finish = 2 ( b. y0 o4 G' S0 |( y}; ) s% o1 v) y) c) @2 l+ O4 c D N
typedef int Pid; 6 c4 y) V0 b/ s Q- Y+ i# O. W! ]9 etypedef int TimeSlice; & L4 @) C4 l: A- K7 O6 J- }, i) ktypedef int Priority; 2 {1 W! J: e/ B% V/ r 5 n0 J/ g: z5 L( `0 Z9 e0 W) Istruct JCB;% O; P: ^. K8 T* a9 u
typedef JCB * JCBptr; * Q" q: f0 [, Y! W5 R( g z( k6 O. q* T5 N6 \, K# G
struct PCB7 u! f3 X+ u2 s! q
{- z, }2 U$ g. `5 {
PCB(JCBptr jcb_) : jcb(jcb_){}& ^7 g, P! }; B
Pid pid;& k$ z- A% b6 u& @
ProcessStatus status; . C/ x2 u% r6 s! b- Z. p Priority priority; & s3 C* m( Q' p/ L4 Q" F" |/ s JCBptr jcb;# [% d: w' Q# f) j6 R7 M& g
}; 7 l' N2 Z3 C. d8 k ! d9 S! e! b, U; ]. z4 {typedef PCB * PCBptr;: u5 V, x _* G
}! Z% z- P6 M8 D: c4 y! q! aenum JobStatus. O* j' y9 {" L) i* j7 V5 A, ], N6 l
{ ) E: Q6 f6 I- O8 D7 }4 I Wait = 0,& Q) F, @2 j1 s; r6 B
Run = 1,+ e4 U! h/ t8 q5 e8 @, d3 |% Y. o
Finish = 2 ; D1 A- N+ V5 R1 e}; . M, _# \! g$ }2 Q. c! b% b9 {! a! S; P5 L& s5 _" L
struct JCB) W& o/ {, d2 D. v; b R/ Q
{ . h7 r' }5 J, v7 U8 |; ]% V std::string job_name;6 ` ~3 e2 {; a2 j: j* f8 Z6 ^
TimeSlice submit_time;$ L5 N9 o/ q) D# {* w
TimeSlice required_running_time;5 a) n2 Q3 p1 R I# _+ Q% w
: Z- A1 `: K3 f) q0 E" J
TimeSlice start_time = -1; 7 d3 ~) O5 h: d9 }3 L TimeSlice finish_time = 0; + ]7 ^6 t$ A$ M: C2 @* y8 a TimeSlice already_run_time = 0; 1 E b$ y! T! x. _, P. D8 H: C2 y" I0 G: M
Resource required_resource;) S2 Z; K; l! o. R0 O6 C2 n
JobStatus job_status;3 Y7 z7 C/ B D* U( y
PCBptr pcb = new PCB(this);. S' c; q7 P1 h: C
}; / M7 G4 I! @- V/ s3 X1 l8 {! i) Y, f) r
5 a) Z/ y) ^0 H: Q7 ]2 O4 z' ?( Pclass Util G& L' [3 U7 `0 i{4 r1 D. I6 M1 `* S: n) F
public: - K' V, h* e( F( W7 _/ ~+ z static inline int getRandom(const int min_val, const int max_val, int match) : t! k6 q! h, F, r O2 ?* E% C {5 {4 K e+ C0 }6 k
srand(time(0) + match); 6 e. b ^1 _6 o1 \9 T9 F/ z return rand() % (max_val - min_val - 1) + min_val; $ }" D) ` `9 S; ~+ g } * N3 ]- u: M4 ` 4 G5 `- B" ?- ]- l static inline bool printJobData(const JCB jcb)" x! o. O2 T" w1 _0 Z, H
{: Z5 s" ~ H( f8 I& X) _/ E& t4 E5 V
TimeSlice start_time = cur_time; # h# }3 z% w/ [/ ?& q1 ^$ z1 J9 U7 g$ p TimeSlice finish_time = start_time + jcb.required_running_time; 5 v. b- {. F1 a! V# ?5 h TimeSlice turnaround_time = finish_time - jcb.submit_time;0 Z" J* I- `# W/ Z2 _7 Z$ o
double power_turnaround_time = turnaround_time / (jcb.required_running_time / 1.0);& p$ B7 j# Z1 N; x2 {! H
v- ^' W" u( f: E
std::cout << "[Data] " << jcb.job_name << " "( M$ V& u/ v G6 V
<< " Start time: " << start_time << " "% g/ k) ?1 F/ Z) v+ c: E% Q/ U5 p2 t
<< " Finish time: " << finish_time << " "5 v* }7 t( _# C. x; L
<< " Turnaround time: " << turnaround_time << " "/ a( A8 Z. b9 u4 E+ k( G
<< " Power turnaround time: " << std::setprecision(2) << power_turnaround_time << std::endl;- R, b8 R' y+ u4 m. M* _8 s: K
0 a4 Q- H' N \+ Y4 H: D$ k ( V* X- D% \, S( T average_turnaround_time += turnaround_time;) o I! y7 J. X+ I7 y3 @& [
average_power_turnaround_time += power_turnaround_time; ) A' F& h; A* ?8 M M* P- m4 s9 S8 T
return true; " }. D0 N u2 E- h }. _/ e" ~7 k9 f- j
7 V. P/ _" s( n: z- v static inline bool printJobData(const JCB jcb, bool isMul) : V3 f! R. _/ O5 J3 j {0 C% n" O' |% L# w6 P( Z
TimeSlice turnaround_time = jcb.finish_time - jcb.submit_time; # E/ l! z+ i. r double power_turnaround_time = turnaround_time / (jcb.required_running_time / 1.0);3 q, Q. s2 a; W* V. ]' u; T# z
4 t' O7 Z" Q9 j M6 c; p l/ J std::cout << "[Data] " << jcb.job_name << " " 6 F1 k/ p, _6 n& d& H << " Finish time: " << jcb.finish_time << " " 4 N; i1 Y" m% Y << " Turnaround time: " << turnaround_time << " "1 r8 ?2 a4 a. l) v; ]! v
<< " Power turnaround time: " << std::setprecision(2) << power_turnaround_time << std::endl; . U+ Y/ c" A5 o- o / i. |& l8 U, B5 w" d1 I% w% r* x. p4 h" g, c3 _
average_turnaround_time += turnaround_time; " `3 o, ]1 N6 B. _8 _0 s average_power_turnaround_time += power_turnaround_time;7 e) s' |" O6 U3 W' z; `4 j7 p
) @4 ]$ `, o' m return true; : j5 y6 s" b; v2 D } / F. F* X$ h# |& u( q! ~! R# @}; " q$ a5 a2 C/ O2 y, W5 m( A! c w1 W9 b: P n
class SchedulingAlgorithm " A2 Z% o* g: b0 n{0 a( I- @) l1 w1 F/ d
public: & M$ G2 e5 B3 b - u5 @% R$ b m# c7 B) c// virtual bool initPCB(std::priority_queue<JCB, std::vector<JCB>, std::less<JCB>> * ready_queue, int process_num) = 0;5 A! E; W* g9 W1 o4 U, f# r
// virtual bool processPCB(JCB & pcb) = 0; " Y- \, O; P `}; ' Q1 I: r) X( g2 K : F5 M7 O% K" o, M. ~5 q" n// First-Come First-Served! c5 c' N/ x0 l/ ]: @; T5 M/ u1 p% T Z
class FCFSSchedulingAlgorithm : public SchedulingAlgorithm7 v1 `# y3 j6 g' N7 h6 ^+ b
{. @; P7 @7 x% D. D
public: # l" G+ H' |/ d5 C: r- F( W0 @! I3 d struct PriorityCmp% T/ O# g; o+ H( j& E1 X" Q9 N
{; q4 B6 l& C3 T" Y
bool operator()(const JCB j1, const JCB j2) / V( |1 |# z% L. [ { & p% I3 M* y5 q return j1.submit_time > j2.submit_time;1 I- V0 y; G" {: d' T) E: r
}/ i j, [# e* e; }- t4 g/ g! j
};# [* T1 T& F$ q+ {. d: B8 j% s
2 q/ C# ^, i* C( I+ g FCFSSchedulingAlgorithm()3 [! w- {1 d, H) O k6 I
{}' i& A% y# L J, ~2 m c
2 M; e: b* B+ ~8 [' m
private:( F) Y; P* d) }4 I6 E
+ N# {' O% s$ K% k( [0 {
1 e: ^3 v7 Y. T1 e
}; $ k1 X1 D! l% ]4 v " B2 _# t) k& G% _3 |9 P// Short Job First , }- V! o) @8 i b% Oclass SJFSchedulingAlgorithm : public SchedulingAlgorithm. |' S: N3 k" R( L* n& w' p& P
{% U) V6 r+ p) `) O4 W ]: [ C
public: 3 w' g* I/ N. s% L; z! ~1 g struct PriorityCmp0 {4 r7 v6 I% k( u& y9 N2 b
{ ' G# f7 `4 ?; y% r8 p bool operator()(const JCB j1, const JCB j2) 0 v4 Q; S0 f: A6 Y { 2 z; S) k; y) P* o! j! F7 B( ?( P return j1.required_running_time > j2.required_running_time; 2 r6 t' j- S! b/ d1 x; o+ |( u+ a } # P) ]( [0 d: x/ {% j }; # J" ~! S [. ?! c1 u. R6 _ 2 Q* h& A, ^4 D- A3 ?8 y& I, |" X SJFSchedulingAlgorithm() 2 d% x- m5 @1 B0 Y4 I( ~ {}; C9 Z0 h3 Z+ B" `7 |