一、实习内容
选择一个调度算法,实现处理器调度。
二、实习目的
在采用多道程序设计的系统中,往往有若干个进程同时处于就绪状态。当就绪进程个数大于处理器数时,就必须依照某种策略来决定哪些进程优先占用处理器。本实习模拟在单处理器情况下的处理器调度,帮助学生加深了解处理器调度的工作。
三、实习题目
本实习有两个题,学生可选择其中的一题做实习。
第一题:设计一个按优先数调度算法实现处理器调度的程序。
[提示]:
(1) 假定系统有五个进程,每一个进程用一个进程控制块PCB来代表,进程控制块的格式为:
进程名 |
指针 |
要求运行时间 |
优先数 |
状态 |
其中,进程名——作为进程的标识,假设五个进程的进程名分别为P1,P2,P3,P4,P5。
指针——按优先数的大小把五个进程连成队列,用指针指出下一个进程的进程控制块的首地址,最后一个进程中的指针为“0”。
要求运行时间——假设进程需要运行的单位时间数。
优先数——赋予进程的优先数,调度时总是选取优先数大的进程先执行。
状态——可假设有两种状态,“就绪”状态和“结束”状态。五个进程的初始状态都为“就绪”,用“R”表示,当一个进程运行结束后,它的状态为“结束”,用“E”表示。
(2) 在每次运行你所设计的处理器调度程序之前,为每个进程任意确定它的“优先数”和“要求运行时间”。
(3) 为了调度方便,把五个进程按给定的优先数从大到小连成队列。用一单元指出队首进程,用指针指出队列的连接情况。例:
队首标志
K2
K1 | P1 | K2 | P2 | K3 | P3 |
| K4 | P4 | K5 | P5 |
| 0 |
| K4 |
| K5 |
|
| K3 |
| K1 |
| 2 |
| 3 |
| 1 |
|
| 2 |
| 4 |
| 1 |
| 5 |
| 3 |
|
| 4 |
| 2 |
| R |
| R |
| R |
|
| R |
| R |
| PCB1 |
| PCB2 |
| PCB3 |
|
| PCB4 |
| PCB5 |
(4) 处理器调度总是选队首进程运行。采用动态改变优先数的办法,进程每运行一次优先数就减“1”。由于本实习是模拟处理器调度,所以,对被选中的进程并不实际的启动运行,而是执行:
优先数-1
要求运行时间-1
来模拟进程的一次运行。
提醒注意的是:在实际的系统中,当一个进程被选中运行时,必须恢复进程的现场,让它占有处理器运行,直到出现等待事件或运行结束。在这里省去了这些工作。
(5) 进程运行一次后,若要求运行时间¹0,则再将它加入队列(按优先数大小插入,且置队首标志);若要求运行时间=0,则把它的状态修改成“结束”(E),且退出队列。
(6) 若“就绪”状态的进程队列不为空,则重复上面(4)和(5)的步骤,直到所有进程都成为“结束”状态。
(7) 在所设计的程序中应有显示或打印语句,能显示或打印每次被选中进程的进程名以及运行一次后进程队列的变化。
(8) 为五个进程任意确定一组“优先数”和“要求运行时间”,启动所设计的处理器调度程序,显示或打印逐次被选中进程的进程名以及进程控制块的动态变化过程。
package System; public class Pcb{ private String name; //进程名 private int time; //进程时间 private int priorty; //优先数; private String status; //状态 private Pcb pointer; //指针 public Pcb(String name, int time, int priorty, String status, Pcb pointer) { this.name = name; this.time = time; this.priorty = priorty; this.status = status; this.pointer = pointer; } public Pcb(){} public String getName() { return name; } public void setName(String name) { this.name = name; } public Pcb getPointer() { return pointer; } public void setPointer(Pcb pointer) { this.pointer = pointer; } public int getTime() { return time; } public void setTime(int time) { this.time = time; } public int getPriorty() { return priorty; } public void setPriorty(int priorty) { this.priorty = priorty; } public String getStatus() { return status; } public void setStatus(String status) { this.status = status; } //显示进程 public void show() { System.out.println("进程" + name + ", 指针->" + pointer.getName() + ", 进程时间=" + time + "," + " 优先数=" + priorty+ ", 状态=" + status ); } }
package System; import java.util.Iterator; import java.util.LinkedList; public class Operate { LinkedList<Pcb> TotalPcb =new LinkedList<Pcb>(); //创建存储PCB的队列 Pcb nullPcb = new Pcb("0", 0, 0, "E", null);//创建一个空PCB对象 //创建PCB进程 public void CreatePcb(int n){ for(int i=1;i<=n;i++){ Pcb pcb=new Pcb(); pcb.setName("p"+i); int priorty=(int)(Math.random()*10); //产生随机优先数 int time=(int)(Math.random()*10)+1; //产生随机时间数 pcb.setPriorty(priorty); pcb.setTime(time); pcb.setPointer(nullPcb); pcb.setStatus("R"); TotalPcb.add(pcb); } } //显示PCB队列 public void showPcb(){ Iterator<Pcb> p=TotalPcb.iterator(); while(p.hasNext()){ Pcb pcb=p.next(); pcb.show(); } System.out.println(); } //删除PCB队列中运行结束的进程 public void remove(){ for(int i=0;i<TotalPcb.size();i++){ if(TotalPcb.get(i).getStatus()=="E"){ System.out.println(TotalPcb.get(i).getName()+"运行结束,退出队列"); TotalPcb.remove(i); } } } //运行后优先数-1,时间-1 public void renew(Pcb p) { if (p.getPriorty()>0) { p.setPriorty(p.getPriorty()-1); //优先数-1 } if (p.getTime()>0) { p.setTime(p.getTime()-1); //时间-1 } if (p.getTime()==0){ p.setStatus("E"); //若时间=0,状态置E } } //队列排序 public void sortPcb() { for(int i=0;i<TotalPcb.size();i++) { for(int j=i+1;j<TotalPcb.size();j++) { if(TotalPcb.get(i).getPriorty()<TotalPcb.get(j).getPriorty()) { //优先数大的先运行 Pcb temp=TotalPcb.get(i); TotalPcb.set(i, TotalPcb.get(j)); TotalPcb.set(j, temp); }else if(TotalPcb.get(i).getPriorty()==TotalPcb.get(j).getPriorty()&&TotalPcb.get(i).getTime()>TotalPcb.get(j).getTime()) { //优先数相同,时间最少的先运行 Pcb temp=TotalPcb.get(i); TotalPcb.set(i, TotalPcb.get(j)); TotalPcb.set(j, temp); } } } //进程指针指向下一个进程 for (int k =0;k<TotalPcb.size()-1; k++) { TotalPcb.get(k).setPointer(TotalPcb.get(k+1)); } //最后一个进程指向为0 if(TotalPcb.size()!=0){ TotalPcb.get(TotalPcb.size()-1).setPointer(nullPcb); } } //进程运行 public void run(){ sortPcb(); //排序 System.out.println("********************************************************"); while(TotalPcb.size()!=0){ System.out.println("运行前进程队列:"); showPcb(); //显示队列 System.out.println(TotalPcb.get(0).getName()+"进程运行,运行时间-1,优先级-1"); renew(TotalPcb.get(0)); System.out.print(TotalPcb.get(0).getName()+"进程运行之后:"); TotalPcb.get(0).show(); remove(); //删除PCB队列中运行结束的进程 sortPcb(); System.out.println("\n运行后进程队列:"); showPcb(); System.out.println("********************************************************"); } System.out.println("所有进程已运行完毕"); } }
package System; import java.util.Scanner; public class Test { public static void main(String[] args) { Scanner input =new Scanner(System.in); Operate op=new Operate(); System.out.print("请输入进程数:"); op.CreatePcb(input.nextInt()); op.run(); } }