Linux:进程调度

简介: Linux:进程调度

进程优先级

查看优先级

在Linux中,进程是有优先级的,我们可以通过指令ps -la来查看:

其中PRI表示priority优先级,在Linux中,优先级的范围是[60, 99]共40个级别。

其中80是优先级的默认值,PRI的值越小,进程的优先级就越高。

Linux之所以把优先级控制在40个,是为了防止有的进程把自己的优先级调的很高,导致一直占用CPU,其他进程得不到资源。这个问题叫做进程饥饿问题。

调整优先级

那么我们要如何调整一个进程的优先级呢?

在刚刚的图示中,在PRI后面还有一个项目,NI,其代表nice值。进程优先级PRINI满足以下公式:

PRI=80+NI

所以我们想要调整进程的优先级PRI,就是要调整nice值。

现在在test.exe可执行程序中执行以下代码:

#include <stdio.h>    
#include <unistd.h>    
#include <stdbool.h>    
    
int main()    
{    
    printf("I am a process, pid = %d\n", getpid());    
    
    while(true)    
    {}    
    
    return 0;                                                                                               
}    

该进程输出自己的PID后,就处于一个死循环状态。

输出结果:

此时进程一直死循环,我们通过ps -la观察一下:

可以看到,现在多了一个PID=23432的进程,也就是我们执行的test.exe,其PRI值为默认值80,现在我们开始修改该进程的nice

  1. 执行top指令,进入资源管理器:

你会进入如下页面:

  1. 按下r键:

r代表你想要改变某个进程的nice,此时白色行上面会出现一个光标,提示你输入想要改变进程的PID

3. 输入PID

输入PID后,现在你就可以改变nice值了:

  1. 输入修改后的NI值:

此处我把nice值设为10,按下回车。

  1. 最后按下q退出管理器

现在我们通过ps -la观察一下:

test.exeNI变成了10,而PRI变成了80 + NI = 90了。


Linux 2.6 内核进程调度队列

现在我们再来看看,进程是如何被调度的,以及优先级是如何起作用的。

此处我以Linux 2.6内核的进程调度队列为例。

CPU中,存在大量的寄存器,比如以下常见的寄存器:

eax/ebx/ecx/edx:用于存储临时数据,比如函数返回值

eds/ecs/fg/gs:段寄存器,区分代码与数据

eip:也就是pc指针,指明当前代码执行到那个位置

浮点数寄存器:浮点数运算

ebp/esp:构建栈区

程序在运行的时候,会存储大量的数据,比如当前执行到哪一行代码,上一个语句运算的结果是什么,函数调用到第几层了,等等。这些数据都存储在CPU的寄存器中。CPU中存储的所有临时数据,叫做硬件上下文

当一个进程从CPU中离开,会把所有的硬件上下文都拷贝走,存储在PCB中,这个过程叫做保护上下文

而当一个进程被再次调度的时候,又会把自己的数据写入CPU中,覆盖原始的寄存器中的数据,这个过程叫做恢复上下文


那么我们再来看看运行队列是如何调度进程的

Linux的运行队列run_queue视图如下:

我们先看到蓝色区域:

其中nr_active表示当前run_queue总共含有几个进程

queue[140]就是运行队列的主体,我们将其划分为两个区间[0, 99]这段区间暂不讲解,[100, 139]这个区间用于放正在排队的进程的PCB.

这个区间刚好有40个元素,而我们的优先级也刚好有40个,也就是说一个优先级的进程对应一个下标。

其实这个数组本质是一个指针数组,类型是task_struct*,即指向PCB类型的指针,每个指针指向一个链表,链表内存储着所有该优先级的进程的PCB

比如优先级为80的进程,都存储在queue[120]指向的链表中

运行队列调度进程的时候,从下标100开始,也就是从高优先级开始调度,然后遍历这个链表,把高优先级的链表执行完,再去执行下一个优先级的链表的进程

bitmap是一个五个元素的int数组,其是一个位图,一个int有32字节,5×32=160,比140大一些。

其用一个位图来表示某个下标对应链表有没有进程,有就是1,没有就是0。此时就可以通过遍历位图,来快速判断一个queue的某个下标位置有没有进程,进而得出要不要去遍历该下标的链表。

那么现在有一个问题,那就是刚刚提到的进程饥饿问题:

假设现在CPU正在执行下标为110的链表的进程,也就是优先级为70的进程,此时刚好又有很多优先级为70的进程进来了,结果CPU一直在执行这个优先级的进程。最后70优先级以后的进程,一直拿不到CPU资源,导致进程饥饿问题。

为了解决该问题,其实run_queue中有两个运行队列,在上图中你会发现,红色区域和蓝色区域的是一模一样的:

这就是两个相同的队列,一个叫做活跃进程队列,一个叫做过期进程队列

active指针指向的队列就是活跃进程队列,expired的指针指向的队列就是过期进程队列

CPU只执行活跃进程队列中的进程,而新来的进程进入过期队列,此时新来的进程就不会影响正在执行的进程,解决了进程饥饿的问题。

当CPU把活跃进程队列的进程执行完后,此时expiredactive指针进行交换,那么活跃进程与过期进程就交换了,此时CPU就去执行新的活跃进程队列。

这样的两个队列轮流执行,一个负责执行,一个负责接受新进程的结构,就解决了进程的饥饿问题

相关文章
|
28天前
|
算法 Linux 调度
深入理解Linux操作系统的进程管理
本文旨在探讨Linux操作系统中的进程管理机制,包括进程的创建、执行、调度和终止等环节。通过对Linux内核中相关模块的分析,揭示其高效的进程管理策略,为开发者提供优化程序性能和资源利用率的参考。
65 1
|
2月前
|
算法 调度 UED
深入理解操作系统:进程调度与优先级队列
【10月更文挑战第31天】在计算机科学的广阔天地中,操作系统扮演着枢纽的角色,它不仅管理着硬件资源,还为应用程序提供了运行的环境。本文将深入浅出地探讨操作系统的核心概念之一——进程调度,以及如何通过优先级队列来优化资源分配。我们将从基础理论出发,逐步过渡到实际应用,最终以代码示例巩固知识点,旨在为读者揭开操作系统高效管理的神秘面纱。
|
16天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
79 13
|
23天前
|
SQL 运维 监控
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
|
30天前
|
存储 算法 调度
深入理解操作系统:进程调度的奥秘
在数字世界的心脏跳动着的是操作系统,它如同一个无形的指挥官,协调着每一个程序和进程。本文将揭开操作系统中进程调度的神秘面纱,带你领略时间片轮转、优先级调度等策略背后的智慧。从理论到实践,我们将一起探索如何通过代码示例来模拟简单的进程调度,从而更深刻地理解这一核心机制。准备好跟随我的步伐,一起走进操作系统的世界吧!
|
8天前
|
Java Linux API
[JavaEE]———进程、进程的数据结构、进程的调度
操作系统,进程任务,PCB,PID,内存指针,文件描述符表,进程的调度,并发编程,状态,优先级,记账信息,上下文
|
1月前
|
运维 监控 Linux
Linux操作系统的守护进程与服务管理深度剖析####
本文作为一篇技术性文章,旨在深入探讨Linux操作系统中守护进程与服务管理的机制、工具及实践策略。不同于传统的摘要概述,本文将以“守护进程的生命周期”为核心线索,串联起Linux服务管理的各个方面,从守护进程的定义与特性出发,逐步深入到Systemd的工作原理、服务单元文件编写、服务状态管理以及故障排查技巧,为读者呈现一幅Linux服务管理的全景图。 ####
|
2月前
|
消息中间件 算法 调度
深入理解操作系统:进程管理与调度
操作系统是计算机系统的核心,负责管理和控制硬件资源、提供用户接口以及执行程序。其中,进程管理是操作系统的重要组成部分,它涉及到进程的创建、调度、同步和通信等方面。本文将深入探讨进程管理的基本概念、进程调度算法以及进程间的同步和通信机制。通过本文的学习,读者将能够更好地理解操作系统的工作原理,并掌握进程管理的基本技能。
54 11
|
2月前
|
算法 调度 UED
深入理解操作系统:进程管理与调度策略
操作系统作为计算机系统的核心,其进程管理和调度策略对于系统性能和用户体验至关重要。本文将通过直观的代码示例和浅显易懂的语言,带领读者了解操作系统如何有效管理进程以及常见的进程调度算法。我们将从进程的基本概念出发,逐步深入到进程状态、进程控制块(PCB)的作用,最后探讨不同的调度算法及其对系统性能的影响。无论您是初学者还是有一定基础的开发者,都能从中获得有价值的信息。
|
2月前
|
负载均衡 算法 调度
深入理解操作系统:进程管理与调度
在数字世界的心脏,操作系统扮演着至关重要的角色。它如同一位精明的指挥家,协调着硬件资源和软件需求之间的和谐乐章。本文将带你走进操作系统的核心,探索进程管理的艺术和调度策略的智慧。你将了解到进程是如何创建、执行和消亡的,以及操作系统如何巧妙地决定哪个进程应该在何时获得CPU的青睐。让我们一起揭开操作系统神秘的面纱,发现那些隐藏在日常计算背后的精妙机制。