操作系统之进程调度——优先权法和轮转法(附上样例讲解)

简介: 多道程序设计中,经常是若干个进程同时处于就绪状态,必须依照某种策略来决定那个进程优先占有处理机。因而引起进程调度。本实验模拟在单处理机情况下的处理机调度问题,加深对进程调度的理解。

要求


一、实验目的



多道程序设计中,经常是若干个进程同时处于就绪状态,必须依照某种策略来决定那个进程优先占有处理机。因而引起进程调度。本实验模拟在单处理机情况下的处理机调度问题,加深对进程调度的理解。


二、实验内容



1.优先权法、轮转法


简化假设


1)进程为计算型的(无I/O)

2)进程状态:ready、running、finish

3)进程需要的CPU时间以时间片为单位确定

2.算法描述

1)优先权法——动态优先权

当前运行进程用完时间片后,其优先权减去一个常数。

2)轮转法


三、流程图



优先权:


20181203114140314.png


轮转法:

20181203114153322.png


分析


?想要完成操作系统算法第一步要弄清楚操作系统相关的专业术语。弄清各个算法的流程和目的要求。才能模拟出相关算法的过程。下面先附上实验的相关要求?


通过本次实验,深刻的理解了操作系统中线程资源的分配方式和进程的调度方式。操作系统实验重在理解每一个算法的意图和目的,那么就选择适当的数据结构模拟过程就可以完成相关算法了。


优先权算法:


1.所有线程的序列核心是围绕优先权的权值大小。并且该优先权的大小会动态的变化,那么我们选区以权值为排序准则的优先队列是处理该结构的最好方法。能够有效的节省空间,算法复杂度。

2.而优先权算法某个线程的结束标识是还需要的时间needtime。所以在随机选区线程的时候和输出状态的时候要判断该线程还需不需要资源。

3.至于状态还有一点很重要的是要即使转换。当进行下一个操作要即使转换上一个线程的状态和下一个线程的状态防止状态混淆。


轮转法:


轮转法强调先进先出的拉链式顺序,而不以其他的权值作为开始/调度的先后顺序,所以普通先进先出的普通队列是解决该算法的最好方法。


. 轮转法和优先权法不一样的是优先权法每次只进一个线程只执行一次。而轮转法是进一个可以执行最多是该线程可轮转的次数/轮转值(可能在中间就完成线程的释放),所以在写程序的时候每次都要判断是否已经轮转。并且到最后还要判断还是否需要调度。如果需要,再抛入队尾。


代码


实现的代码(java):


import java.util.ArrayDeque;
import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.Scanner;
public class priority {
  static pcb pcb[];
  public static void main(String[] args) {
    // TODO 自动生成的方法存根
    System.out.println("创建线程数量:");
    Scanner sc=new Scanner(System.in);
    int n=sc.nextInt();
    pcb=new pcb[n];
    for(int i=0;i<n;i++)
    {
      pcb[i]=new pcb(i+1,"ready");
    }
    System.out.println("是否采用优先权?Y/N");
    String s=sc.next();
    if(s.equals("Y"))
    {
      priority();
    }
    else
    {
      lunzhuan();
    }
    //printstatus();
  }
  private static void lunzhuan() {
    Queue<pcb>q1=new ArrayDeque<>();
    for(int i=0;i<pcb.length;i++)
    {
      int lunhzun=(int) (Math.random()*4)+1+(int)(Math.random()*2);//不能为0
      int needtime=(int)(Math.random()*8)+1;
      pcb[i].lunhzuan=lunhzun;
      pcb[i].needalltime=needtime;  
      q1.add(pcb[i]);
    }
    while(!q1.isEmpty())
    {
      pcb team=q1.poll();
      int time=0;//占用cpu时间片树
      while(time<team.lunhzuan)
      {
        if(team.needalltime<=0)
        {break;}
        time++;
        team.needalltime-=1;
        team.status="running";
        printluzhuan();
        System.out.println();
      }
      if(team.needalltime<=0) {team.status="finish";}
      else
      {
        team.status="ready";
        q1.add(team);
      }
    }
    printluzhuan();
  }
  private static void printluzhuan() {
    for(int i=0;i<pcb.length;i++)
    {
      System.out.println("threadid:"+pcb[i].id+"  "+pcb[i].status+"  needtime:"+pcb[i].needalltime+"  lunzhuan:"+pcb[i].lunhzuan);
    } 
  }
  private static void priority() {//优先权
    Queue<pcb>q1=new PriorityQueue<pcb>(com);
    for(int i=0;i<pcb.length;i++)
    {
      int proty=(int) (Math.random()*15);
      int needtime=(int)(Math.random()*4)+1;
      pcb[i].proty=proty;
      pcb[i].needalltime=needtime;
      q1.add(pcb[i]);
    } 
//    for(int i=0;i<pcb.length;i++)
//    {
//      q1.add(pcb[i]);
//    }
    while(!q1.isEmpty())
    {
      pcb team=q1.poll();
      team.needalltime-=1;
      team.proty-=3;
      team.status="running";
      printstatus();
      System.out.println();
      if(team.needalltime>0)
      {
        team.status="ready";
        q1.add(team); 
      }
      else
        team.status="finish";
    }
    printstatus();
  }
  private static void printstatus() {
    // TODO 自动生成的方法存根
    for(int i=0;i<pcb.length;i++)
    {
      System.out.println("threadid:"+pcb[i].id+"  "+pcb[i].status+"  needtime:"+pcb[i].needalltime+"  priority:"+pcb[i].proty);
    }   
  }
  static Comparator<pcb> com=new Comparator<pcb>() {
    @Override
    public int compare(pcb o1, pcb o2) {
      // TODO 自动生成的方法存根
      return o2.proty-o1.proty;
    }
  }; 
  static class pcb
  { 
    int id;//进程id
    String status;//进程状态
    int proty;
    int needalltime;//需要总时间
    int lunhzuan;//轮转时间树
    public pcb(int id, String status) {
      // TODO 自动生成的构造函数存根
      this.id=id;
      this.status=status;
    } 
  }
}


数据打印



20181203115256905.png


2:轮转法


20181203115324327.png


声明:这只是本人的个人理解,如果又纰漏,还请大神指出!?


目录
相关文章
|
算法 Linux 调度
深入理解Linux操作系统的进程管理
本文旨在探讨Linux操作系统中的进程管理机制,包括进程的创建、执行、调度和终止等环节。通过对Linux内核中相关模块的分析,揭示其高效的进程管理策略,为开发者提供优化程序性能和资源利用率的参考。
520 32
|
弹性计算 运维 资源调度
使用阿里云操作系统控制台巧解调度抖动
阿里云操作系统控制台是一站式云服务器管理平台,提供性能监控、故障诊断、日志分析、安全管理和资源调度等功能。用户可实时查看CPU、内存等使用情况,快速定位并解决调度抖动等问题。智能诊断工具自动生成优化建议,简化运维流程,降低技术门槛。尽管部分功能仍在优化中,但整体上显著提升了云服务器管理的效率和稳定性。
364 15
使用阿里云操作系统控制台巧解调度抖动
|
缓存 运维 前端开发
阿里云操作系统控制台:高效解决性能瓶颈与抖动之进程热点追踪
遇到“进程性能瓶颈导致业务异常”等多项业务痛点时,提供高效解决方案,并展示案例。
|
12月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
341 0
|
监控 搜索推荐 开发工具
2025年1月9日更新Windows操作系统个人使用-禁用掉一下一些不必要的服务-关闭占用资源的进程-禁用服务提升系统运行速度-让电脑不再卡顿-优雅草央千澈-长期更新
2025年1月9日更新Windows操作系统个人使用-禁用掉一下一些不必要的服务-关闭占用资源的进程-禁用服务提升系统运行速度-让电脑不再卡顿-优雅草央千澈-长期更新
3341 2
2025年1月9日更新Windows操作系统个人使用-禁用掉一下一些不必要的服务-关闭占用资源的进程-禁用服务提升系统运行速度-让电脑不再卡顿-优雅草央千澈-长期更新
|
C语言 开发者 内存技术
探索操作系统核心:从进程管理到内存分配
本文将深入探讨操作系统的两大核心功能——进程管理和内存分配。通过直观的代码示例,我们将了解如何在操作系统中实现这些基本功能,以及它们如何影响系统性能和稳定性。文章旨在为读者提供一个清晰的操作系统内部工作机制视角,同时强调理解和掌握这些概念对于任何软件开发人员的重要性。
|
Linux 调度 C语言
深入理解操作系统:从进程管理到内存优化
本文旨在为读者提供一次深入浅出的操作系统之旅,从进程管理的基本概念出发,逐步探索到内存管理的高级技巧。我们将通过实际代码示例,揭示操作系统如何高效地调度和优化资源,确保系统稳定运行。无论你是初学者还是有一定基础的开发者,这篇文章都将为你打开一扇了解操作系统深层工作原理的大门。
232 4
|
存储 算法 调度
深入理解操作系统:进程调度的奥秘
在数字世界的心脏跳动着的是操作系统,它如同一个无形的指挥官,协调着每一个程序和进程。本文将揭开操作系统中进程调度的神秘面纱,带你领略时间片轮转、优先级调度等策略背后的智慧。从理论到实践,我们将一起探索如何通过代码示例来模拟简单的进程调度,从而更深刻地理解这一核心机制。准备好跟随我的步伐,一起走进操作系统的世界吧!
|
算法 调度 开发者
深入理解操作系统:进程与线程的管理
在数字世界的复杂编织中,操作系统如同一位精明的指挥家,协调着每一个音符的奏响。本篇文章将带领读者穿越操作系统的幕后,探索进程与线程管理的奥秘。从进程的诞生到线程的舞蹈,我们将一起见证这场微观世界的华丽变奏。通过深入浅出的解释和生动的比喻,本文旨在揭示操作系统如何高效地处理多任务,确保系统的稳定性和效率。让我们一起跟随代码的步伐,走进操作系统的内心世界。
217 2

推荐镜像

更多