无畏并发: Rust Mutex的基本使用

并发是很多编程语言避不开的一块主要内容,主打一个无畏并发的Rust自然也面临这样的挑战。Rust中的Mutex提供了强大的同步原语,确保共享数据的线程安全,这篇文章中,我们会探讨Mutex的使用,从基础的用法到一些高阶内容。
在这里插入图片描述


1. Rust中的互斥锁

Mutex作为Rust中的一种互斥锁,它一次只允许一个线程访问共享资源。

  • 提供了一种同步可变数据访问的机制,防止数据竞争、确保线程安全;

  • 位于Rust标准库std::sync::Mutex中,支持对共享可变状态的安全并发访问。

  • 互斥锁允许一个线程在访问共享数据时获取锁,而其他线程必须等待该线程释放锁后才能访问共享数据,从而确保数据的安全性和一致性。


2. 基本用法体验

下面以使用Mutex保护共享数据为例演示Mutex的基础用法。

use std::{sync::Mutex, thread};
fn main() {let counter = Mutex::new(0);let handle = thread::spawn(move || {let mut value = counter.lock().unwrap();*value += 1;});handle.join().expect("thread panicked!");let value = counter.lock().unwrap();println!("Counter:{}", *value);
}

这段代码演示了Mutex的基本用法,但它是无法通过编译的,尽管如此,这里还是对上述代码进行解释一下:

  1. let counter = Mutex::new(0);

    • 在这行代码中,创建了一个名为 counterMutex,其中包含一个整数值初始为 0。这个 Mutex 将用于保护共享的整数值,以确保线程安全访问。
  2. let handle = thread::spawn(move || { ... });

    • 使用 thread::spawn 创建一个新线程,并在该线程中执行一个闭包。这个闭包中的内容将在新线程中运行。
    • move 关键字用于将 counter 的所有权转移给闭包,确保闭包可以在新线程中使用 counter
  3. let mut val = counter.lock().unwrap();

    • 在闭包中,通过调用 lock() 方法获取 counter 的锁,并使用 unwrap() 处理可能的锁获取失败的情况。这里的 val 是一个 MutexGuard,它允许我们安全地访问被 Mutex 保护的数据。
  4. *val += 1;

    • 在获取了 counter 的锁之后,对共享数据进行递增操作。
  5. handle.join().expect("thread panicked!");

    • 等待新线程执行完毕。join() 方法会阻塞当前线程,直到新线程执行完毕。
    • 如果新线程发生了 panic,expect() 方法会打印指定的错误信息。
  6. let val = counter.lock().unwrap();

    • 在主线程中,再次获取 counter 的锁,以确保安全地访问共享数据。
  7. println!("Counter:{}", *val);

    • 打印最终的共享数据值。

那么问题来了,为什么无法通过编译:

  • 在这段代码在编译阶段会出现 borrow of moved value 的错误,因为我们在 thread::spawn 的闭包中移动了 counter,而后又在闭包外部尝试再次使用它。

  • 为了解决这个问题,我们需要使用 Arc (原子引用计数) 来在多线程之间安全地共享 counter

use std::{sync::{Arc, Mutex},thread,
};
fn main() {let counter = Arc::new(Mutex::new(0));let counter_clone = Arc::clone(&counter);let handle = thread::spawn(move || {let mut value = counter_clone.lock().unwrap();*value += 1;});match handle.join() {Ok(_) => {let value = counter.lock().unwrap();println!("Counter: {}", *value);}Err(_) => {println!("Thread panicked!");}}
}

修复之后的代码中,使用了Arcmutex进行了调整,通过使用 Arc 来创建一个引用计数的 Mutex,并在闭包中使用 Arc 的克隆。这样可以确保在多线程环境中安全地共享 counter,避免了 borrow of moved value 的问题。同时使用match表达式进一步对错误进行了处理。


3. 多个锁的使用

有时,我们可能需要独立保护多个共享资源。 Rust 允许您使用多个锁来实现这一点:

use std::{sync::{Arc, Mutex},thread,
};fn main() {let cnt1 = Arc::new(Mutex::new(0));let cnt2 = Arc::new(Mutex::new(0));let cnt1_clone = Arc::clone(&cnt1);let handle1 = thread::spawn(move || {let mut value = cnt1_clone.lock().unwrap();*value += 1;});let cnt2_clone = Arc::clone(&cnt2);let handle2 = thread::spawn(move || {let mut value2 = cnt2_clone.lock().unwrap();*value2 += 1;});match handle1.join() {Ok(_) => {let value1 = cnt1.lock().unwrap();println!("cnt1:{}", *value1);}Err(_) => {println!("Thread panicked!");}}match handle2.join() {Ok(_) => {let value2 = cnt2.lock().unwrap();println!("cnt2:{}", *value2);}Err(_) => {println!("Thread panicked!");}}
}

这段代码演示了如何使用 ArcMutex 在 Rust 中实现多线程并发访问共享数据:

  1. 首先,通过 use 关键字引入了需要使用的标准库中的一些模块,包括 ArcMutexthread

  2. main 函数中,创建了两个计数器 cnt1cnt2,它们分别被包装在 ArcMutex 中。这样做是为了确保在多线程环境中安全地访问这两个计数器。

  3. 使用 Arc::clone(&cnt1)Arc::clone(&cnt2) 创建了两个 Arc 的克隆,分别赋值给 cnt1_clonecnt2_clone。这样做是为了将计数器的所有权移动到新的线程中。

  4. 通过 thread::spawn 创建了两个线程 handle1handle2,分别对 cnt1_clonecnt2_clone 所指向的计数器进行操作。在每个线程中,首先获取了计数器的锁,然后对计数器的值进行加一操作。

  5. 使用 handle1.join()handle2.join() 分别等待两个线程的执行结果。如果线程执行成功,就获取相应计数器的锁并打印计数器的值;如果线程执行失败(比如发生了 panic),则打印出相应的错误信息。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/16204.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024电工杯B题平衡膳食食谱的优化设计及评价原创论文分享

大家好,从昨天肝到现在,终于完成了2024电工杯数学建模B题的完整论文啦。 实在精力有限,具体的讲解大家可以去讲解视频: 给大家看一下目录吧: 目录 摘 要: 10 一、问题重述 14 二.问题分析 …

Python学习---基于HTTP的服务端基础框架搭建案例

整体功能: 1 创建框架构建相关的文件夹 2 创建app,模块文件 3 在 app模块文件中创建application函数(用于处理请求) 4 将request_handler()中的处理逻辑交由app模块的application函数完成 5 app模块的 application函数返回响应报文 6 在application 文件夹中创建一个…

淘工厂订单导出自动化工具

目录 下载安装与运行 主要目的 其他工具的弊端 本工具的优势 视频演示 下载新版后的注意事项 支持的导出项 什么叫一单多拍 常见问题 如何实现快捷登录 导出卡住时如何操作 如何精确导出 下载安装与运行 下载、安装与运行 语雀 主要目的 导出订单信息&#xf…

知识分享:隔多久查询一次网贷大数据信用报告比较好?

随着互联网金融的快速发展,越来越多的人开始接触和使用网络贷款。而在这个过程中,网贷大数据信用报告成为了评估借款人信用状况的重要依据。那么,隔多久查询一次网贷大数据信用报告比较好呢?接下来随小易大数据平台小编去看看吧。 首先&…

某钢铁企业数字化转型规划案例(114页PPT)

案例介绍: 该钢铁企业的数字化转型案例表明,数字化转型是钢铁企业应对市场竞争、提高生产效率、降低成本、优化资源配置和降低能耗排放的重要手段。通过引入先进的技术和管理理念,加强员工培训和人才引进,企业可以成功实现数字化…

欢乐钓鱼大师游戏攻略:自动钓鱼技巧!

《欢乐钓鱼大师》是一款极具趣味性和挑战性的钓鱼模拟游戏,为玩家提供了一个体验钓鱼乐趣的虚拟世界。从湖泊到河流,再到广袤的海洋,游戏中的各种钓场让人流连忘返。无论是新手钓友,还是经验丰富的老钓手,都可以在游戏…

k8s集群中pod的容器资源限制和三种探针

一、资源限制 总结: requests表示创建pod时预留的资源,limits表示pod能够使用资源的最大值。requests值可以被超,limits值不能超过,如果是内存使用超过limits会触发oom然后杀掉进程,如果是cpu超过limits会压缩cpu的使用…

20.SkyWalking

一.简介 SkyWalking用于应用性能监控、分布式链路跟踪、诊断: 参考连接如下: https://github.com/apache/skywalking https://skywalking.apache.org/docs/ 二.示例 通过官网连接进入下载页面:https://archive.apache.org/dist/skywalkin…

揭秘章子怡成功之路:她是如何征服世界的?

章子怡的演艺生涯可谓是一部传奇❗❗❗ 从一个普通工人家庭的女孩,到如今的国际巨星 她的每一步都充满了努力和汗水 她的舞蹈基础为她日后的演艺事业奠定了坚实的基础 而她对戏剧和电影的热爱更是让她在演艺道路上不断前行 从《我的父亲母亲》到《卧虎藏龙》&…

代码随想录|Day55|动态规划 part15|● 392.判断子序列 ● 115.不同的子序列

392.判断子序列 class Solution: def isSubsequence(self, s: str, t: str) -> bool: dp [[0] * (len(t) 1) for _ in range(len(s) 1)] for i in range(1, len(s) 1): for j in range(1, len(t) 1): if s[i - 1] t[j - 1]: dp[i][j] dp[i - 1][j - 1] 1 else: dp[i…

【UE5.1 角色练习】06-角色发射火球-part2

目录 效果 步骤 一、火球生命周期 二、添加可被伤害的NPC 三、添加冲量 在上一篇(【UE5.1 角色练习】06-角色发射火球-part1)基础上继续实现角色发射火球相关功能 效果 步骤 一、火球生命周期 为了防止火球没有命中任何物体而一直移动下去&#…

【全开源】赛事报名系统源码(Fastadmin+ThinkPHP和Uniapp)

基于FastadminThinkPHP和Uniapp开发的赛事报名系统,包含个人报名和团队报名、成绩查询、成绩证书等。 构建高效便捷的赛事参与平台 一、引言:赛事报名系统的重要性 在举办各类赛事时,一个高效便捷的报名系统对于组织者和参与者来说都至关重…

WebService的wsdl详解

webservice服务的wsdl内容详解,以及如何根据其内容编写调用代码 wsdl示例 展示一个webservice的wsdl,及调用这个接口的Axis客户端 wsdl This XML file does not appear to have any style information associated with it. The document tree is shown…

编译aosp刷入pixel 真机得问题记录

编译aosp要做什么(ubuntu下编译问题相对少) 需要vmware并且已经安装了ubuntu镜像系统 直接切换到root 避免后期避免麻烦 参考地址 https://mp.weixin.qq.com/s/yJp3ijIxykiMmNVYr2V1nQ apt install git //安装git sudo apt install git//给git设置用户…

mysql事务 事务并发问题 隔离级别 以及原理

mysql事务 简介:事务是一组操作的集合,它是一个不可分割的工作单位,事务会把所有的操作作为一个整体一起向系统提交或撤销操作请求,即这些操作要么同时成功,要么同时失败。 事务四大特性 原子性(Atomici…

.哈希表.

哈希 哈希表:将大而复杂的数据映射到紧凑的区间内。分为:①存储结构 (离散化是特殊的哈希,之前讲的离散化是严格保序的 映射到区间上是连续递增的) 哈希不保序,这里讲的是一般的哈希 弊端:若…

Linux(四)

Linux(四) shell脚本shell脚本开发过程创建创建.sh文件编写.sh文件添加执行的权限 chmod 777 1.sh运行 shell中注释shell中变量用户自定义变量 (尽量大写)位置参数即命令行参数预定义变量环境变量 shell中程序和语句说明性语句功能性语句echo 输出read 键…

网上打印试卷的步骤是什么

对于学生和家长来说,打印试卷是日常学习中的一项重要需求。那么,如何在网上方便地打印试卷呢?下面,就让我来为您介绍琢贝云打印的试卷打印步骤。 一、选择琢贝云打印的原因 支持多种文件格式打印,包括图片、PPT、PDF、…

每日百万交易的支付系统,如何设置JVM堆内存大小?

每日百万交易的支付系统,如何设置JVM堆内存大小? 1、支付背景的引入2、支付的核心业务流程3、每日百万交易支付系统的压力在哪里?4、支付系统每秒钟需要处理多少笔支付单5、每个支付订单处理需要耗时多久6、每个支付订单大概需要多大的内存空间7、每秒发起的支付请求对内存的…

手撕C语言题典——消失的数字

目录 前言 一,思路 1)排序查找 2)数据求和,依次减去中值 3) 异或 二,异或的代码实现 前言 依旧是一道力扣上的题,通过不同思路的不同时间复杂度来分析,让我们看看有什么不同。 面试题 17…