Rust所有权和Move关键字使用和含义讲解,以及Arc和Mutex使用

Rust 所有权规则

一个值只能被一个变量所拥有,这个变量被称为所有者。
一个值同一时刻只能有一个所有者,也就是说不能有两个变量拥有相同的值。所以对应变量赋值、参数传递、函数返回等行为,旧的所有者会把值的所有权转移给新的所有者,以便保证单一所有者的约束。
当所有者离开作用域,其拥有的值被丢弃,内存得到释放。
这三条规则很好理解,核心就是保证单一所有权。其中第二条规则讲的所有权转移是 Move 语义,Rust 从 C++ 那里学习和借鉴了这个概念。

第三条规则中的作用域(scope)指一个代码块(block),在 Rust 中,一对花括号括起来的代码区就是一个作用域。举个例子,如果一个变量被定义在 if {} 内,那么 if 语句结束,这个变量的作用域就结束了,其值会被丢弃;同样的,函数里定义的变量,在离开函数时会被丢弃。

所有权规则,解决了谁真正拥有数据的生杀大权问题,让堆上数据的多重引用不复存在,这是它最大的优势。 但是,它也有一个缺点,就是每次赋值、参数传递、函数返回等行为,都会导致旧的所有者把值的所有权转移给新的所有者,这会导致一些性能上的问题。

Move关键字

Rust 是一门以安全性著称的系统编程语言,它允许程序员高效地进行并发编程。在 Rust 中,线程是一种重要的并发原语,通过标准库提供的 std::thread 模块,我们可以轻松地创建和管理线程。而 Move 闭包是一种特殊的闭包,它可以在创建时传递外部变量的所有权,使得在多线程环境中传递数据更加灵活和高效。

Rust 中的线程

在 Rust 中,线程是一种独立的执行流,它允许程序在不同的执行路径上同时运行。Rust 的线程模型采用了“共享状态,可变状态”(Shared State, Mutable State)的方式,这意味着多个线程可以访问同一个数据,但需要通过锁(Lock)来保证数据的安全性。

创建线程:在 Rust 中,我们可以使用 std::thread::spawn 函数来创建一个新的线程。下面是一个简单的例子:

use std::thread;fn main() {let handle = thread::spawn(|| {println!("Hello from the new thread!");});handle.join().unwrap();
}

在上述示例中,我们调用 thread::spawn 函数创建了一个新的线程,并在该线程中打印一条信息。注意,thread::spawn 函数接受一个闭包作为参数,闭包中的代码会在新线程中执行。

线程间通信
在多线程编程中,线程间通信是一个重要的问题。在 Rust 中,我们可以使用 std::sync 模块提供的同步原语来实现线程间的安全通信。常见的同步原语包括 Mutex(互斥锁)和 Arc(原子引用计数)等。

下面是一个使用 Mutex 实现线程安全计数的例子:

use std::sync::{Arc, Mutex};
use std::thread;fn main() {let counter = Arc::new(Mutex::new(0));let mut handles = vec![];for _ in 0..10 {let counter = Arc::clone(&counter);let handle = thread::spawn(move || {let mut num = counter.lock().unwrap();*num += 1;});handles.push(handle);}for handle in handles {handle.join().unwrap();}println!("Result: {}", *counter.lock().unwrap());
}

在上述示例中,我们创建了一个 Mutex 来包装计数器变量 counter,以实现线程安全的计数。在每个线程中,我们通过 counter.lock().unwrap() 获取 Mutex 的锁,然后通过 *num += 1 修改计数器的值。在修改完成后,锁会自动释放。

Move 闭包

Rust 中的闭包有三种形式:Fn、FnMut 和 FnOnce。其中,FnOnce 是最特殊的一种,它可以消耗捕获的变量,并且只能被调用一次。这种特性使得 FnOnce 闭包可以在创建时携带外部变量的所有权,并在闭包内使用这些变量。

在线程中使用 Move 闭包:
在多线程编程中,有时我们希望在线程创建时将一些数据传递给新线程,并且希望新线程拥有这些数据的所有权,这时就可以使用 Move 闭包。

下面是一个使用 Move 闭包的例子:

use std::thread;fn main() {let data = vec![1, 2, 3, 4, 5];let handle = thread::spawn(move || {for num in data {println!("Number: {}", num);}});handle.join().unwrap();
}

在上述示例中,我们创建了一个 data 向量,并在 thread::spawn 函数中使用 move 关键字将 data 向量的所有权转移给了新线程。这样,新线程就拥有了 data 向量的所有权,可以在闭包中访问和使用它。

需要注意的是,使用 Move 闭包时要特别小心数据的所有权转移。如果在闭包外部继续使用了数据,可能会导致编译错误或运行时错误: 

使用 Arc 和 Move 闭包

在某些情况下,我们希望在多个线程中共享数据,并且某些线程需要拥有数据的所有权。这时,可以结合使用 Arc 和 Move 闭包来实现。

下面是一个使用 Arc 和 Move 闭包的例子:

use std::sync::{Arc, Mutex};
use std::thread;fn main() {let data = Arc::new(Mutex::new(vec![1, 2, 3, 4, 5]));let handles: Vec<_> = (0..5).map(|i| {let data = Arc::clone(&data);thread::spawn(move || {let mut data = data.lock().unwrap();data[i] += 1;})}).collect();for handle in handles {handle.join().unwrap();}println!("Result: {:?}", *data.lock().unwrap());
}

在上述示例中,我们创建了一个 data 向量,并将它包装在 Arc 和 Mutex 中以实现线程安全共享。然后,我们使用 map 方法创建了5个线程,并在每个线程中修改 data 向量的一个元素。通过使用 Move 闭包和 Arc,每个线程都拥有了 data 向量的所有权,可以在闭包中修改它。

多线程与 Move 闭包的应用场景

多线程和 Move 闭包在 Rust 中有着广泛的应用场景,尤其是在并发处理和性能优化方面。以下是一些常见的应用场景:

并行计算:多线程可以同时执行独立的任务,提高计算速度和性能。
并发服务器:服务器需要同时处理多个客户端请求,多线程可以使服务器更高效地处理并发请求。
数据处理:在数据处理任务中,多线程可以同时处理不同的数据块,加速数据处理过程。

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

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

相关文章

后疫情时代CS保研沉思录暨2023年个人保研经验贴

个人情况 正如古话所说&#xff0c;最适合你的才是最好的。因此这里先贴上个人基本情况&#xff0c;用作参考。 如果你的个人情况与我相近&#xff0c;则有更强的参考作用。如果情况相差较大&#xff0c;也可以姑且引为例子来研究。 学校层次&#xff1a;中流至末流211 专业…

R语言中的常用数据结构

目录 R对象的基本类型 R对象的属性 R的数据结构 向量 矩阵 数组 列表 因子 缺失值NA 数据框 R的数据结构总结 R语言可以进行探索性数据分析&#xff0c;统计推断&#xff0c;回归分析&#xff0c;机器学习&#xff0c;数据产品开发 R对象的基本类型 R语言对象有五…

EKS-1.26 创建ingress-nginx绑定elb暴露服务

1. 创建集群 &#xff08;跳过不介绍&#xff09; 2. 创建Ingress-Nginx服务 部署项目地址【点我跳转】 推荐自定义部署 可绑定acm证书什么的自己属性 对应集群版本推荐阵列 https://github.com/kubernetes/ingress-nginx 修改下面的下载文件版本 Download the deploy.yaml…

Xxxxxx

数据库 1&#xff0c;B树与B树区别 1&#xff0c;B树每个节点存ID与其他数据字段&#xff0c;B非叶子结点&#xff0c;只存ID&#xff0c;叶子结点存完整数据 好处&#xff1a;每个层级B树&#xff0c;可以存储更多的额数据&#xff0c;层级更少&#xff0c;更扁平&#xff…

视觉大模型--deter的深入理解

但对于transformer用于目标检测领域的开创性模型&#xff0c;该模型言简意赅&#xff0c;但是但从论文理解&#xff0c;有很多细节都不清楚&#xff0c;尤其是解码器的query和二分图匹配(Bipartite Matching)和匈牙利算法(Hungarian Algorithm)相关&#xff0c;本文将根据代码详…

GLP-1药物固相合成法-载体树脂及层析填料

摘要&#xff1a;在生物医药GLP-1药物制备领域不仅可提供高稳定性载体树脂&#xff0c;还可根据客户需求&#xff0c;合成定制化载体&#xff08;如预接氨基酸固相合成载体、特殊溶胀度或基团负载量的载体、负载特殊基团的载体、清除树脂等&#xff09;。同时&#xff0c;海普专…

使用TCP协议就一定零丢包了吗?

简述数据包发送流程 为了简化模型&#xff0c;我们把中间的服务器给省略掉&#xff0c;假设这是个端到端的通信。且为了保证消息的可靠性&#xff0c;它们之间用的是TCP协议进行通信。 为了发送数据包&#xff0c;两端首先会通过三次握手&#xff0c;建立TCP连接。 一个数据包&…

Grafana+Promethues配置RocketMQ监控

背景 接前文&#xff0c;Promethues已经配置完毕&#xff0c;下面通过导入的Grafana的面板来配置RocketMQ监控页面 Dashboard 这里我们直接使用Grafana现成的面板配置 node_exporter&#xff1a;https://grafana.com/grafana/dashboards/1860 rocketmq_exporter的dashboar…

【蓝桥杯-枚举模板题】

蓝桥杯-枚举模板题 滑雪课程设计 P3650新的家乡 P8587枚举子集 B3622Air Cownditioning P9011 滑雪课程设计 P3650 核心的思路是把数据规定在[i,i17]里&#xff0c;不够的补&#xff0c;过大的减。枚举i以求最少的钱。 #include<bits/stdc.h> using namespace std;int n…

C#实现只保存2天的日志文件

文章目录 业务需求代码运行效果 欢迎讨论&#xff01; 业务需求 在生产环境中&#xff0c;控制台窗口不便展示出来。 为了在生产环境中&#xff0c;完整记录控制台应用的输出&#xff0c;选择将其输出到文件中。 但是&#xff0c;存储所有输出的话会占用很多空间&#xff0c;…

Chrome 设置在新窗口中打开链接(已登录google账号版)

Chrome的链接默认是在原标签页中打开的&#xff0c;如果要在新窗口中打开&#xff0c;需要自己自行设置&#xff0c;在此&#xff0c;针对已经登录google账号的chrome浏览器怎么进行设置进行说明。 一、点击登录图标->更多设置 二、选择其他设置->在新窗口中打开搜索结果…

15、Scalable Diffusion Models with Transformers

简介 官网 DiT&#xff08;Diffusuion Transformer&#xff09;将扩散模型的 UNet backbone 换成 Transformer&#xff0c;并且发现通过增加 Transformer 的深度/宽度或增加输入令牌数量&#xff0c;具有较高 Gflops 的 DiT 始终具有较低的 FID&#xff08;~2.27&#xff09;…