信号量与管程

2023-11-05 02:40
文章标签 信号量 管程

本文主要是介绍信号量与管程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

我们知道,在并发领域内,需要关注分工、同步与互斥,针对分工问题,就是将任务拆解,分配给多个线程执行,而在多线程执行的过程中,需要解决线程之间的协作与互斥问题进而保证并发安全。那么解决这类问题的方案是什么呢?没错就是信号量和管程。

信号量

简介

信号量的概念是由荷兰计算机科学家Edsger W. Dijkstra在1960年引入的。Dijkstra引入了P(Proberen,荷兰语中的"try")和V(Verhogen,荷兰语中的"increment")这两个操作,并使用它们来解决各种同步问题,如著名的哲学家进餐问题。

Dijkstra最初引入信号量的目的是为了管理稀缺的计算机资源,如打印机或磁带驱动器。但随着时间的推移,信号量被广泛应用于各种场景中,成为并发编程中的基石。

信号量有两种常见类型:

  • 二进制信号量:其值只能为0或1,类似于互斥锁,常用于资源的互斥访问。
  • 计数信号量:其值可以为任何非负整数,常用于管理有限的资源集,如线程池中的线程数量或数据库连接池中的连接数量。

实现原理

信号量模型比较简单,它由一个计数器、一个等待队列和三个方法组成,即如下图所示:
在这里插入图片描述

信号量模型维护一个计数器来决定进入临界区的线程数,init方法则是初始化计数器大小,P操作则是将计数器-1,V操作则是把计数器+1,信号量的运转流程如下图所示:
在这里插入图片描述

demo演示

package com.markus.concurrent;import java.util.concurrent.Semaphore;/*** @author: markus* @date: 2023/8/19 2:21 PM* @Description: 信号量demo* @Blog: https://markuszhang.com* It's my honor to share what I've learned with you!*/
public class SemaphoreDemo {public static void main(String[] args) throws InterruptedException {Semaphore semaphore = new Semaphore(1);int count = 0;ShareObject shareObject = new ShareObject(semaphore, count);Thread threadA = new Thread(() -> {for (int i = 0; i < 100_000; i++) {// 线程安全shareObject.increment();}});Thread threadB = new Thread(() -> {for (int i = 0; i < 100_000; i++) {shareObject.increment();}});threadA.start();threadB.start();threadA.join();threadB.join();System.out.println(shareObject.getCount());}
}class ShareObject {private Semaphore semaphore;private int count;public ShareObject(Semaphore semaphore, int count) {this.semaphore = semaphore;this.count = count;}public void increment() {try {semaphore.acquire();// 临界区 非原子性操作,如果不做同步互斥控制,会造成并发不安全的情况count += 1;} catch (InterruptedException e) {e.printStackTrace();} finally {semaphore.release();}}public void unsafeIncrement() {count++;}public int getCount() {return this.count;}
}

管程

简介

管程的概念是在1970s由Edsger W. Dijkstra、C.A.R. Hoare和Per Brinch Hansen等计算机科学家独立提出的。其中,C.A.R. Hoare的论文“Monitors: An Operating System Structuring Concept”特别影响深远,他详细描述了管程的结构和特性,并提出了条件变量的概念。

管程的提出旨在简化并发编程中复杂的同步问题,提供一个更高级和更结构化的同步方法。与信号量相比,管程通常更易于理解和使用,因为它将同步机制与数据结构紧密集成,并自动处理互斥。

许多现代编程语言和操作系统都提供了原生或类似管程的支持。例如,Java的synchronized关键字和内置的对象锁提供了管程的基本功能(互斥),而Object类的wait(), notify(), 和notifyAll()方法则实现了条件变量的功能。

上面提到两个关键组件:

  • 互斥访问:管程确保其方法在任何时候都只能被一个线程执行。
  • 条件变量:允许线程等待特定条件成立或通知其他等待的线程条件已经改变。这是管程中的核心部分,常常用于线程间同步。

实现原理

与信号量不同,管程是将共享变量、同步队列封装了起来,并在此基础上增加了条件变量及其等待队列,管程模型如下图所示:

MESA模型

需要一提的是:上图是管程MESA模型的实现,还有另外一种模型可以实现管程:Hoare模型,MESA模型和Hoare模型的核心区别就在于:

  • 在Hoare模型中,当一个线程在条件变量上执行signal操作来唤醒另一个线程时,控制权会立即被传递给被唤醒的线程。这意味着,唤醒的线程立刻获得管程的锁并开始执行,执行signal操作的线程将被暂停,直到被唤醒的线程释放锁或进入等待状态。
  • 在MESA模型中,当线程被唤醒时,它并不立即重新获得管程的锁。相反,它被放入一个就绪队列,等待重新获得锁。这种行为可能会导致所谓的“叫醒后等待”(wakeup-wait)的情况,即一个被唤醒的线程可能在获得锁之前需要等待其他线程。

MESA模型的优势在于它通常更容易实现,并且可以减少上下文切换的数量。

Java选择MESA模型来实现其内置的管程(Monitor)机制主要基于以下几个原因:

  • 实现简便性:MESA模型简化了唤醒和调度的过程。在Hoare模型中,当一个线程执行signal操作时,它必须将锁传递给被唤醒的线程,这可能导致额外的上下文切换和调度复杂性。而在MESA模型中,执行signal操作的线程可以继续执行,直到它自然地释放锁。
  • 减少上下文切换:如前所述,MESA模型可以减少不必要的上下文切换,因为执行signal的线程不必立即放弃执行权。
  • 预测性:在多处理器系统上,MESA模型可以提供更好的性能和预测性。由于线程不需要立即传递控制权,这有助于在多处理器环境中实现更有效的锁缓存和减少锁迁移。
  • 假唤醒的处理:MESA模型天然地支持处理假唤醒(spurious wakeups)。线程在被唤醒后会重新检查等待的条件,这样可以确保即使因为假唤醒而被唤醒,线程也不会执行不应该执行的代码。

尽管MESA模型引入了所谓的"叫醒后等待"(wakeup-wait)的现象,但由于上述优点,Java开发者认为它是一个更好的选择。这也是为什么Java的Object.wait()Object.notify()/notifyAll()方法的行为与MESA模型相吻合。

Java 内置的管程方案(synchronized)使用简单,synchronized 关键字修饰的代码块,在编译期会自动生成相关加锁和解锁的代码,但是仅支持一个条件变量;而 Java SDK 并发包实现的管程支持多个条件变量,不过并发包里的锁,需要开发人员自己进行加锁和解锁操作。

demo演示

package com.markus.concurrent;/*** @author: markus* @date: 2023/8/19 3:12 PM* @Description:* @Blog: https://markuszhang.com* It's my honor to share what I've learned with you!*/
public class Synchronized4MonitorDemo {public static void main(String[] args) throws InterruptedException {ShareInteger shareInteger = new ShareInteger(0);Thread threadA = new Thread(() -> {for (int i = 0; i < 100_000; i++) {try {shareInteger.increment();} catch (InterruptedException e) {e.printStackTrace();}}});Thread threadB = new Thread(() -> {for (int i = 0; i < 100_000; i++) {try {shareInteger.increment();} catch (InterruptedException e) {e.printStackTrace();}}});threadA.start();threadB.start();Thread.sleep(2000);System.out.println("主线程将count设置为501");shareInteger.setCount(501);// 等待两个线程执行完threadA.join();threadB.join();// 打印最终的加和结果System.out.println(shareInteger.getCount());}
}class ShareInteger {private int count;public ShareInteger(int count) {this.count = count;}public void increment() throws InterruptedException {synchronized (this) {while (count <= 500) {System.out.println(Thread.currentThread().getName() + " 被阻塞");this.wait();System.out.println(Thread.currentThread().getName() + " 被唤醒");}count += 1;}}public void setCount(int count) {synchronized (this) {this.count = count;if (count > 500) {// 唤醒所有等待count>500条件的线程this.notifyAll();}}}public int getCount() {return count;}
}

信号量与管程的对比

管程和信号量都是用于处理并发问题的同步原语,但它们具有不同的特点和使用方法。下面是管程和信号量的优劣对比以及它们的使用场景:

  • 管程:
    • 优点:
      • 封装性:管程提供了良好的封装性,因为它将数据和对数据的操作包含在一个单一的结构或对象中。这使得管程在面向对象的环境中特别有用。
      • 简介性:管程自动处理锁的获取和释放,使得代码更简洁且易于理解。
      • 条件变量支持:管程内部的条件变量提供了一种强大的机制,允许线程在特定条件下等待或被唤醒。
    • 缺点:
      • 灵活性:相对于信号量,管程可能在某些特定场景下不那么灵活。
    • 使用场景:
      • 需要结构化并发控制的情境,尤其是在面向对象的设计中。
      • 当需要使用条件变量来处理复杂的同步条件时。
  • 信号量:
    • 优点:
      • 灵活性:信号量为并发控制提供了极大的灵活性。它可以用于实现互斥、同步,以及各种资源计数场景
      • 广泛性:信号量是很多操作系统和并发库的基石,它有着广泛的应用
    • 缺点:
      • 易出错:由于信号量的灵活性,使用它的代码可能更容易出错。例如,忘记释放信号量或不正确的信号量使用可能导致死锁
      • 缺乏封装:信号量不提供封装数据和操作的机制,可能导致数据不一致或竞争条件。
    • 使用场景:
      • 用于实现互斥锁和其他锁类型。
      • 管理有限资源的数量,如线程池中的线程或数据库连接。
      • 实现复杂的同步场景,如生产者-消费者问题。(这里并没有管程实现简单,并且使用不当还会出错)

管程与信号量在Java中都有相应的实现,基于不同的场景应用不同的模型,并不是说谁好谁不好,只能说在某种场景下,谁比谁更合适,例如实现一个限流器,信号量就优于管程;实现一个阻塞队列,管程就优于信号量

其他同步工具

下面罗列下其他同步工具,做一些简要介绍,后续会单拉出几篇文章做详细解释。

CAS

原子操作,它检查当前值是否与预期值匹配,如果匹配,则使用新值更新它。

读拷贝更新

一种同步机制,允许读取操作无锁并发地执行,而更新操作通过延迟回收机制避免与读取操作冲突。

读写锁

一种锁机制,允许多个读者并发访问,但在写入时保证独占访问。

障碍同步

一种同步原语,使一组线程在继续执行之前等待所有线程都到达某个点。

总结

总结起来,管程和信号量都是并发控制的核心工具,各自带有其独特的特点和使用方法。管程,通过其结构化和封装的特性,为复杂的同步问题提供了简单、直观的解决方案,尤其适用于面向对象的环境中。它们强调了数据和对数据的操作之间的紧密结合,确保数据的完整性和安全性。而信号量,作为一种更基础且灵活的同步原语,能够用于广泛的场景,从基本的互斥到复杂的协调任务。虽然信号量提供了更大的灵活性,但这种灵活性也可能带来更高的错误风险。因此,在选择适当的并发工具时,开发者需要根据特定的问题和需求来权衡。不管如何,了解这两个工具的工作方式及其优劣势是任何希望深入并发编程的开发者的基础任务。

参考

https://time.geekbang.org/column/intro/100023901?tab=catalog

这篇关于信号量与管程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/346862

相关文章

Linux多线程——POSIX信号量与环形队列版本之生产消费模型

文章目录 POSIX信号量POSIX的操作初始化销毁等待信号量(申请资源)发布信号量(放下资源) 环形队列之生产消费模型 POSIX信号量 POSIX信号量和System V信号量是不同的标准 但是实现的功能是一样的,都是为了解决同步的问题 我们说信号量指的就是资源的数量 在生产者与消费者模型里面,生产者与消费者所认为的资源是不同的 生产者认为空间是资源,因为每次都要

信号与信号量的区别[转]

信号量(Semaphore),有时被称为信号灯,是在多环境下使用的一种设施,是可以用来保证两个或多个关键代码段不被并发调用。在进入一个关键代码段之前,线程必须获取一个信号量;一旦该关键代码段完成了,那么该线程必须释放信号量。其它想进入该关键代码段的线程必须等待直到第一个线程释放信号量。为了完成这个过程,需要创建一个信号量VI,然后将Acquire Semaphore VI以及Release Se

使用信号量实现一个限流器:C++实战指南

使用信号量实现一个限流器:C++实战指南 在现代软件开发中,限流器(Rate Limiter)是一种常用的技术,用于控制系统的请求速率,防止系统过载。信号量(Semaphore)是一种强大的同步原语,可以用于实现限流器。本文将详细介绍如何在C++中使用信号量实现一个限流器,并提供完整的代码示例和详细的解释。 什么是限流器? 限流器是一种控制系统请求速率的机制,确保在单位时间内处理的请求数量不

C++ 信号量

信号量: 在生产者消费者模型中,对任务数量的记录就可以使用信号量来做。当信号量的值小于0时,工作进程或者线程就会阻塞,等待物品到来。当生产者生产一个物品,会将信号量值加1操作 sem_post(&sem)。 这是会唤醒在信号量上阻塞的进程或者线程sem_wait(&sem),它们去争抢物品。 信号量广泛用于进程或线程间的同步和互斥,信号量本质上是⼀个非负的整数计数器,它被用来控制对公共资源的访

Java并发线程 共享模型之管程 5

1. 生产者消费者 package cn.itcast.testcopy;import cn.itcast.n2copy.util.Sleeper;import lombok.extern.slf4j.Slf4j;import java.util.LinkedList;/*** ClassName: Test21* Package: cn.itcast.testcopy* Descript

【嵌入式】内存未对齐导致程序崩溃(铺获信号量SIGBUS,数值7)

背景 嵌入式平台上,和A组配合,需要把A组提供的二进制文件在调用A组提供接口时传入,因为有多个bin文件,自测的时候选择了其中一个,运行正常。递交给qa测试了。然后qa反馈必现崩溃。懵了。复现的时候还用的之前的bin文件,无法复现。最后看信号量数值和打印日志判断是在调用接口的地方,然后对了下长度,发现奇数。而自己用的偶数大小的bin文件。然后修改4字节对齐后正常了。 问题现象 日志打印提示:

【Linux】消息队列信号量

目录 消息队列 原理 接口 指令 信号量 概念 对于信号量理论的理解 信号量的操作  信号量的指令 消息队列 原理 消息队列提供了一个从一个进程向另外一个进程发送一个数据块的方法,每个数据块都有一个类型。对消息队列的的管理也是先描述,再组织! 接口 我们发现,消息队列和共享内存的接口极其相似,消息队列的属性也保存在ipc_perm结构体中,这个结构以中的第一

FreeRTOS线程同步1---信号量

目录 二值信号量 二值信号量相关API函数 一般使用方法为 1.创建二值信号量 2.在一个任务中置为二值释放信号 3.在另一个任务中获取信号 计数信号量 计数型信号量相关 API 函数 使用方法 1.创建计数信号量 2.释放计数信号量 3.获得信号量的当前值 4.释放信号量 互斥信号量 互斥信号量常用API函数 使用方法         信号量是一种解决同

java 信号量Semaphore的使用

java 信号量Semaphore的使用 信号量是一种计数器,用来保护一个或者多个共享资源的访问。 信号量的使用: (1)如果一个线程要访问一个共享资源,他必须先获得信号量。如果信号量的内部计数器大于0,信号量将减1,然后允许访问这个共享资源。计数器大于0意味着又可以使用的资源,因此线程讲被允许使用其中一个资源。 (2)如果信号量等于0,信号将将会把线程植入休眠直到计数器大于0.计数器等于

【Linux修行路】进程通信——消息队列、信号量

目录 ⛳️推荐 一、消息队列 1.1 实现原理 1.2 消息队列接口 1.2.1 msgget——创建、获取一个消息队列 1.2.2 msgctl——释放消息队列、获取消息队列属性 1.2.3 msgsnd——发送数据 1.2.4 msgrcv——从消息队列中检索数据块 1.3 消息队列的指令操作 二、信号量 2.1 数据不一致问题、互斥、临界资源、临界区 2.2 理解