LeetCode-1117. H2O 生成(多线程)

2024-06-03 13:32

本文主要是介绍LeetCode-1117. H2O 生成(多线程),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

LeetCode 题目描述

现在有两种线程,氢 oxygen 和氧 hydrogen,你的目标是组织这两种线程来产生水分子。

存在一个屏障(barrier)使得每个线程必须等候直到一个完整水分子能够被产生出来。

氢和氧线程会被分别给予 releaseHydrogen 和 releaseOxygen 方法来允许它们突破屏障。

这些线程应该三三成组突破屏障并能立即组合产生一个水分子。

你必须保证产生一个水分子所需线程的结合必须发生在下一个水分子产生之前。

换句话说:

  • 如果一个氧线程到达屏障时没有氢线程到达,它必须等候直到两个氢线程到达。
  • 如果一个氢线程到达屏障时没有其它线程到达,它必须等候直到一个氧线程和另一个氢线程到达。

书写满足这些限制条件的氢、氧线程同步代码。

示例 1:

输入: "HOH"
输出: "HHO"
解释: "HOH" 和 "OHH" 依然都是有效解。

示例 2:

输入: "OOHHHH"
输出: "HHOHHO"
解释: "HOHHHO", "OHHHHO", "HHOHOH", "HOHHOH", "OHHHOH", "HHOOHH", "HOHOHH" 和 "OHHOHH" 依然都是有效解。

限制条件:

  • 输入字符串的总长将会是 3n, 1 ≤ n ≤ 50;
  • 输入字符串中的 “H” 总数将会是 2n;
  • 输入字符串中的 “O” 总数将会是 n。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/building-h2o
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

题解

整体思考思路为:

  • 2 个线程并发执行
  • 2 个线程可执行的数量不同(限制线程数量)
  • 2 个线程等待状态互相制约(当前执行的线程需要生成一个水分子后后放行下一组线程)

1.Semaphore + CyclicBarrier

思路:使用信号量控制 2 个线程的访问数量,使用 CyclicBarrier 控制三三成组的执行。

考虑:CyclicBarrier 比较重量级。

class H2O {// 信号量 保证 H2/0 线程执行等待状态,即每次只有 2 个 H 线程、1 个 O 线程可执行private final Semaphore h2 = new Semaphore(2, false);private final Semaphore o = new Semaphore(1, false);// 屏障 ,保证线程三三成组执行private final CyclicBarrier barrier = new CyclicBarrier(3);public H2O() {}public void hydrogen(Runnable releaseHydrogen) throws InterruptedException {h2.acquire();try {barrier.await();} catch (BrokenBarrierException e) {throw new InterruptedException(e.getMessage());}// releaseHydrogen.run() outputs "H". Do not change or remove this line.releaseHydrogen.run();h2.release();}public void oxygen(Runnable releaseOxygen) throws InterruptedException {o.acquire();try {barrier.await();} catch (BrokenBarrierException e) {throw new InterruptedException(e.getMessage());}// releaseOxygen.run() outputs "O". Do not change or remove this line.releaseOxygen.run();o.release();}
}

2. Semaphore + AtomicInteger

思路:使用信号量控制 2 个线程的访问数量,使用 AtomicInteger(CAS) 控制三三成组的执行。

class H2O {// 信号量 保证 H2/0 线程执行等待状态,即每次只有 2 个 H 线程、1 个 O 线程可执行private final Semaphore h2 = new Semaphore(2, false);private final Semaphore o = new Semaphore(1, false);// 屏障 ,保证线程三三成组执行private final AtomicInteger barrier = new AtomicInteger();public H2O() {}public void hydrogen(Runnable releaseHydrogen) throws InterruptedException {h2.acquire();// releaseHydrogen.run() outputs "H". Do not change or remove this line.releaseHydrogen.run();barrier.getAndIncrement();resetBarrier();}public void oxygen(Runnable releaseOxygen) throws InterruptedException {o.acquire();// releaseOxygen.run() outputs "O". Do not change or remove this line.releaseOxygen.run();barrier.getAndIncrement();resetBarrier();}private void resetBarrier() {if (barrier.compareAndSet(3, 0)) { h2.release(2);o.release();}}
}

参考

  • 我的提交记录

这篇关于LeetCode-1117. H2O 生成(多线程)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

浅析Rust多线程中如何安全的使用变量

《浅析Rust多线程中如何安全的使用变量》这篇文章主要为大家详细介绍了Rust如何在线程的闭包中安全的使用变量,包括共享变量和修改变量,文中的示例代码讲解详细,有需要的小伙伴可以参考下... 目录1. 向线程传递变量2. 多线程共享变量引用3. 多线程中修改变量4. 总结在Rust语言中,一个既引人入胜又可

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

详解Java中如何使用JFreeChart生成甘特图

《详解Java中如何使用JFreeChart生成甘特图》甘特图是一种流行的项目管理工具,用于显示项目的进度和任务分配,在Java开发中,JFreeChart是一个强大的开源图表库,能够生成各种类型的图... 目录引言一、JFreeChart简介二、准备工作三、创建甘特图1. 定义数据集2. 创建甘特图3.

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

AI一键生成 PPT

AI一键生成 PPT 操作步骤 作为一名打工人,是不是经常需要制作各种PPT来分享我的生活和想法。但是,你们知道,有时候灵感来了,时间却不够用了!😩直到我发现了Kimi AI——一个能够自动生成PPT的神奇助手!🌟 什么是Kimi? 一款月之暗面科技有限公司开发的AI办公工具,帮助用户快速生成高质量的演示文稿。 无论你是职场人士、学生还是教师,Kimi都能够为你的办公文

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n