MK
摩柯社区 - 一个极简的技术知识社区
AI 面试

进程间的资源竞争与分配策略

2023-12-115.7k 阅读

进程间资源竞争的本质

在操作系统中,进程是资源分配和调度的基本单位。多个进程同时运行时,它们可能会竞争有限的系统资源,如 CPU 时间、内存空间、I/O 设备等。这种资源竞争是进程间相互制约的一种表现,它源于资源的有限性和进程对资源需求的多样性。

以 CPU 资源为例,现代操作系统通常采用多任务处理机制,允许多个进程并发执行。然而,CPU 在某一时刻只能执行一个进程的指令。因此,多个进程都希望获得 CPU 时间来推进自己的执行。这就导致了进程间对 CPU 资源的竞争。

同样,内存空间也是有限的。每个进程都需要一定的内存来存储其代码、数据和堆栈等。当多个进程同时运行且所需内存总量超过系统可用内存时,就会发生内存资源的竞争。

对于 I/O 设备,如硬盘、打印机等,它们通常是共享资源。不同进程可能在不同时间需要使用这些设备进行数据读写或打印等操作,从而引发 I/O 资源的竞争。

资源竞争可能会引发一些问题,例如死锁。当一组进程中的每个进程都在等待其他进程释放其所需要的资源,而这些进程又都不会主动释放自己已占有的资源时,就会形成死锁。死锁会导致这些进程都无法继续执行,浪费系统资源,甚至可能导致整个系统瘫痪。

资源分配策略概述

为了有效地管理进程间的资源竞争,操作系统需要采用合理的资源分配策略。这些策略旨在在满足进程资源需求的同时,尽量避免死锁等问题的发生,提高系统的整体性能和资源利用率。

资源分配策略通常可以分为以下几类:

  1. 静态分配策略:在进程创建时,一次性分配给该进程所需的全部资源。进程在运行过程中不会再请求新的资源,直到其运行结束并释放所有已分配的资源。这种策略的优点是简单直观,能够避免死锁的发生,因为进程获得了所有资源后就不会再因资源不足而等待其他进程释放资源。然而,它的缺点也很明显,由于进程在整个运行期间可能并不会同时使用所有分配的资源,这会导致资源的浪费,降低系统资源利用率。
  2. 动态分配策略:进程在运行过程中根据需要动态地请求和释放资源。这种策略更加灵活,能够提高资源的利用率,但同时也增加了死锁发生的风险。因此,在采用动态分配策略时,需要配合相应的死锁预防、避免或检测与解除机制。

常见资源分配策略详解

先来先服务(FCFS, First - Come, First - Served)

  1. 基本原理: 先来先服务是一种简单直观的资源分配策略。当有多个进程请求相同资源时,操作系统按照进程请求资源的先后顺序进行分配。最早请求资源的进程将首先获得资源。

  2. 实现方式: 在实现上,操作系统可以维护一个请求队列。当进程请求资源时,将其加入队列尾部。当资源可用时,从队列头部取出进程并分配资源。

以下是一个简单的 Python 示例模拟 FCFS 资源分配:

class Resource:
    def __init__(self):
        self.available = True

    def allocate(self):
        if self.available:
            self.available = False
            return True
        return False

    def release(self):
        self.available = True


class Process:
    def __init__(self, pid, arrival_time):
        self.pid = pid
        self.arrival_time = arrival_time


def fcfs(resource, processes):
    request_queue = []
    for process in processes:
        request_queue.append(process)
    while request_queue:
        process = request_queue.pop(0)
        if resource.allocate():
            print(f"Process {process.pid} allocated resource at time {process.arrival_time}")
            # 模拟进程使用资源
            resource.release()
            print(f"Process {process.pid} released resource")
        else:
            print(f"Resource not available for Process {process.pid} at time {process.arrival_time}, added to queue")
            request_queue.append(process)


if __name__ == "__main__":
    res = Resource()
    p1 = Process(1, 0)
    p2 = Process(2, 1)
    p3 = Process(3, 2)
    processes = [p1, p2, p3]
    fcfs(res, processes)
  1. 优点与缺点: 优点:算法简单,实现容易,公平性较好,按照请求顺序分配资源,不会偏袒任何进程。 缺点:对于长进程友好,而对短进程不利。如果一个长进程先请求资源,它会占用资源较长时间,导致后面的短进程等待时间过长,降低了系统的整体响应速度。

短作业优先(SJF, Shortest Job First)

  1. 基本原理: 短作业优先策略是根据进程预计运行时间的长短来分配资源。当有多个进程请求资源时,操作系统优先分配资源给预计运行时间最短的进程。这种策略的目的是减少进程的平均等待时间,提高系统的吞吐量。

  2. 实现方式: 操作系统需要维护一个包含进程预计运行时间信息的列表或队列。当资源可用时,从列表中选择预计运行时间最短的进程进行资源分配。在实际实现中,确定进程的预计运行时间可能比较困难,通常可以根据进程过去的运行经验或用户提供的估计值来近似。

以下是一个简单的 SJF 资源分配模拟代码(假设已知进程运行时间):

class Resource:
    def __init__(self):
        self.available = True

    def allocate(self):
        if self.available:
            self.available = False
            return True
        return False

    def release(self):
        self.available = True


class Process:
    def __init__(self, pid, run_time):
        self.pid = pid
        self.run_time = run_time


def sjf(resource, processes):
    sorted_processes = sorted(processes, key=lambda p: p.run_time)
    for process in sorted_processes:
        if resource.allocate():
            print(f"Process {process.pid} allocated resource, run time: {process.run_time}")
            # 模拟进程使用资源
            resource.release()
            print(f"Process {process.pid} released resource")


if __name__ == "__main__":
    res = Resource()
    p1 = Process(1, 5)
    p2 = Process(2, 3)
    p3 = Process(3, 8)
    processes = [p1, p2, p3]
    sjf(res, processes)
  1. 优点与缺点: 优点:能够有效减少进程的平均等待时间,提高系统吞吐量。对于短进程来说,能够快速获得资源并完成运行,提高了用户体验。 缺点:需要预先知道进程的运行时间,这在实际中往往很难准确获取。此外,如果不断有短进程进入系统,长进程可能会被无限期推迟,导致饥饿现象。

优先级调度

  1. 基本原理: 优先级调度策略为每个进程分配一个优先级。当有多个进程请求资源时,操作系统优先分配资源给优先级最高的进程。进程的优先级可以根据多种因素确定,如进程类型(系统进程通常优先级较高)、进程的紧急程度、进程预计运行时间等。

  2. 实现方式: 操作系统需要维护一个优先级队列。当进程请求资源时,根据其优先级将其插入到合适的位置。当资源可用时,从队列头部取出优先级最高的进程分配资源。

以下是一个简单的优先级调度模拟代码:

class Resource:
    def __init__(self):
        self.available = True

    def allocate(self):
        if self.available:
            self.available = False
            return True
        return False

    def release(self):
        self.available = True


class Process:
    def __init__(self, pid, priority):
        self.pid = pid
        self.priority = priority


def priority_scheduling(resource, processes):
    priority_queue = []
    for process in processes:
        # 简单插入排序维护优先级队列
        inserted = False
        for i in range(len(priority_queue)):
            if process.priority > priority_queue[i].priority:
                priority_queue.insert(i, process)
                inserted = True
                break
        if not inserted:
            priority_queue.append(process)
    while priority_queue:
        process = priority_queue.pop(0)
        if resource.allocate():
            print(f"Process {process.pid} allocated resource, priority: {process.priority}")
            # 模拟进程使用资源
            resource.release()
            print(f"Process {process.pid} released resource")


if __name__ == "__main__":
    res = Resource()
    p1 = Process(1, 3)
    p2 = Process(2, 5)
    p3 = Process(3, 2)
    processes = [p1, p2, p3]
    priority_scheduling(res, processes)
  1. 优点与缺点: 优点:可以根据进程的重要性和紧急程度合理分配资源,确保关键进程能够及时获得资源运行。 缺点:如果不采取适当的措施,低优先级进程可能会长期得不到资源,导致饥饿现象。此外,如何合理确定进程的优先级也是一个挑战,过高或过低的优先级设置都可能影响系统性能。

时间片轮转(RR, Round - Robin)

  1. 基本原理: 时间片轮转主要用于 CPU 资源分配。操作系统将 CPU 时间划分为固定长度的时间片,每个进程轮流获得一个时间片来执行。当时间片用完后,即使进程尚未完成,也会被剥夺 CPU 使用权,排到就绪队列末尾等待下一次分配时间片。

  2. 实现方式: 操作系统维护一个就绪队列,将所有就绪进程放入队列。每次从队列头部取出一个进程,分配一个时间片让其运行。时间片结束后,将该进程放回队列尾部。

以下是一个简单的时间片轮转模拟代码:

class Process:
    def __init__(self, pid, burst_time):
        self.pid = pid
        self.burst_time = burst_time
        self.remaining_time = burst_time


def round_robin(processes, time_quantum):
    ready_queue = processes.copy()
    total_time = 0
    while ready_queue:
        process = ready_queue.pop(0)
        if process.remaining_time <= time_quantum:
            total_time += process.remaining_time
            process.remaining_time = 0
            print(f"Process {process.pid} completed in {total_time} units")
        else:
            total_time += time_quantum
            process.remaining_time -= time_quantum
            ready_queue.append(process)


if __name__ == "__main__":
    p1 = Process(1, 10)
    p2 = Process(2, 5)
    p3 = Process(3, 8)
    processes = [p1, p2, p3]
    time_quantum = 3
    round_robin(processes, time_quantum)
  1. 优点与缺点: 优点:公平性好,每个进程都能在一定时间内获得 CPU 时间,不会出现某个进程长期占用 CPU 的情况。响应性较好,能快速响应交互式进程的请求。 缺点:如果时间片设置过小,进程上下文切换频繁,会增加系统开销;如果时间片设置过大,RR 算法会退化为 FCFS 算法,对短进程不利。

死锁相关的资源分配策略

死锁预防

  1. 破坏死锁的必要条件: 死锁的发生需要同时满足四个必要条件:互斥条件、占有并等待条件、不可剥夺条件和循环等待条件。死锁预防策略就是通过破坏这些条件中的一个或多个来避免死锁的发生。

    • 破坏互斥条件:某些资源本身具有互斥性,如打印机,无法改变其互斥特性。但对于一些软件资源,可以通过共享方式来破坏互斥条件。例如,对于可重入代码,可以允许多个进程同时访问。
    • 破坏占有并等待条件:可以采用静态分配策略,即进程在创建时一次性获得所需的全部资源。或者要求进程在请求新资源之前,先释放已占有的所有资源。
    • 破坏不可剥夺条件:当一个进程占有某些资源并请求新资源而得不到满足时,操作系统可以剥夺该进程已占有的部分资源,分配给其他进程。这种方式适用于一些可剥夺资源,如 CPU 时间。
    • 破坏循环等待条件:可以对资源进行编号,规定进程只能按照资源编号递增的顺序请求资源。这样可以避免形成资源请求环路,从而破坏循环等待条件。
  2. 优缺点: 优点:能够从根本上避免死锁的发生。 缺点:某些预防策略可能会降低系统资源利用率和进程并发度。例如,静态分配策略会导致资源浪费,剥夺资源可能会使进程执行出现异常。

死锁避免

  1. 银行家算法: 银行家算法是一种经典的死锁避免算法。该算法基于这样一个思想:操作系统在每次资源分配前,先检查此次分配是否会导致系统进入不安全状态。如果不会,则进行分配;否则,拒绝分配。

银行家算法需要以下数据结构: - Available:表示系统中各类资源的可用数量。 - Max:表示每个进程对各类资源的最大需求。 - Allocation:表示每个进程当前已分配到的各类资源数量。 - Need:表示每个进程还需要的各类资源数量,Need[i][j] = Max[i][j] - Allocation[i][j]

算法步骤如下: - 寻找一个进程 i,其 Need[i] <= Available。如果找到,则表示该进程可以获得所需资源并运行结束。 - 假设进程 i 运行结束并释放资源,更新 AvailableAvailable = Available + Allocation[i]。 - 重复上述步骤,直到所有进程都能运行结束(系统处于安全状态),或者找不到满足条件的进程(系统处于不安全状态)。

以下是一个简单的银行家算法 Python 实现:

def is_safe(available, max_need, allocation):
    need = [[max_need[i][j] - allocation[i][j] for j in range(len(available))] for i in range(len(max_need))]
    finish = [False] * len(max_need)
    work = available.copy()
    found = True
    while found:
        found = False
        for i in range(len(max_need)):
            if not finish[i] and all(need[i][j] <= work[j] for j in range(len(available))):
                work = [work[j] + allocation[i][j] for j in range(len(available))]
                finish[i] = True
                found = True
    return all(finish)


def request_resources(available, max_need, allocation, process, request):
    if all(request[j] <= need[process][j] for j in range(len(available))) and all(request[j] <= available[j] for j in range(len(available))):
        available = [available[j] - request[j] for j in range(len(available))]
        allocation[process] = [allocation[process][j] + request[j] for j in range(len(available))]
        need = [[max_need[i][j] - allocation[i][j] for j in range(len(available))] for i in range(len(max_need))]
        if is_safe(available, max_need, allocation):
            return True, available, allocation
        else:
            available = [available[j] + request[j] for j in range(len(available))]
            allocation[process] = [allocation[process][j] - request[j] for j in range(len(available))]
            return False, available, allocation
    return False, available, allocation


if __name__ == "__main__":
    available = [3, 3, 2]
    max_need = [[7, 5, 3], [3, 2, 2], [9, 0, 2], [2, 2, 2], [4, 3, 3]]
    allocation = [[0, 1, 0], [2, 0, 0], [3, 0, 2], [2, 1, 1], [0, 0, 2]]
    process = 1
    request = [1, 0, 2]
    result, available, allocation = request_resources(available, max_need, allocation, process, request)
    if result:
        print("Request Granted, System is in safe state")
    else:
        print("Request Denied, System would be in unsafe state")
  1. 优缺点: 优点:能够在不影响系统并发度的前提下避免死锁,比死锁预防策略更加灵活。 缺点:需要预先知道每个进程对资源的最大需求,并且算法执行开销较大,每次资源分配都需要进行复杂的安全性检查。

死锁检测与解除

  1. 死锁检测: 死锁检测算法定期检查系统是否存在死锁。常用的死锁检测算法基于资源分配图算法。资源分配图由进程节点和资源节点组成,边表示进程对资源的请求或资源对进程的分配关系。

算法步骤如下: - 简化资源分配图。寻找既不阻塞也不孤立的进程节点,删除该节点及其相关边。重复此步骤,直到无法继续简化。 - 如果最终资源分配图中所有节点都被删除,则系统不存在死锁;否则,存在死锁,图中剩余节点构成死锁环。

  1. 死锁解除: 一旦检测到死锁,需要采取措施解除死锁。常见的解除方法有:

    • 资源剥夺法:从一个或多个死锁进程中剥夺足够数量的资源,分配给其他进程,以打破死锁环。
    • 撤销进程法:强制撤销一个或多个死锁进程,释放它们占有的资源,从而解除死锁。可以选择撤销优先级最低的进程,或者选择撤销运行时间最短的进程,以减少系统损失。
  2. 优缺点: 优点:不需要预先采取复杂的预防或避免措施,在死锁发生概率较低的系统中,开销相对较小。 缺点:死锁发生后会造成一定的系统资源浪费和进程执行中断,检测和解除算法也需要一定的系统开销。

综合资源分配策略的应用

在实际操作系统中,往往不会单纯使用一种资源分配策略,而是综合多种策略来满足不同场景和需求。

例如,在处理 CPU 资源分配时,可以结合时间片轮转和优先级调度。对于交互式进程,给予较高优先级并分配适当的时间片,以保证其快速响应;对于后台批处理进程,优先级较低,按照时间片轮转方式获得 CPU 时间。

在内存资源分配方面,可以采用动态分配策略,并结合分页和分段管理技术。根据进程的需求动态分配内存页面或段,同时通过虚拟内存技术,将暂时不用的内存数据交换到磁盘,以提高内存利用率。

对于 I/O 资源,可以根据设备类型和进程请求特点,采用先来先服务或优先级调度策略。对于打印机等独占设备,采用先来先服务保证公平性;对于磁盘 I/O,可以根据进程的优先级或 I/O 请求的紧急程度进行调度。

同时,为了避免死锁,操作系统通常会综合运用死锁预防、避免和检测解除机制。在系统初始化或进程创建阶段,采用死锁预防策略,如资源编号规则等;在资源分配过程中,结合银行家算法等死锁避免机制;定期进行死锁检测,一旦发现死锁,及时采取解除措施。

通过综合运用这些资源分配策略,操作系统能够在复杂的多进程环境下,有效地管理资源竞争,提高系统性能、资源利用率和稳定性,为用户提供高效可靠的计算环境。