Rust循环引用与多线程并发
循环引用与自引用
循环引用的概念
循环引用指的是两个或多个对象之间相互持有对方的引用。在 Rust 中,由于所有权和生命周期的严格约束,直接创建循环引用通常会导致编译失败。例如:
// 错误的循环引用示例
struct Node {next: Option<Box<Node>>,
}fn create_cycle() {let n1 = Box::new(Node { next: None });let n2 = Box::new(Node { next: Some(n1) }); // 编译错误n1.next = Some(n2); // 编译错误
}
在这个例子中,尝试创建一个简单的双向链表,但由于所有权转移问题,编译器会报错。
自引用结构体的实现
自引用结构体是指一个结构体内部包含对自身实例的引用。这种结构常用于实现树形数据结构或其他需要递归引用的场景。
use std::rc::{Rc, Weak};struct Node {value: i32,parent: Option<Weak<Rc<Node>>>,children: Vec<Rc<Node>>,
}impl Node {fn new(value: i32) -> Self {Node {value,parent: None,children: Vec::new(),}}fn add_child(&mut self, child: Rc<Node>) {self.children.push(child.clone());child.parent = Some(Rc::downgrade(&self));}
}
使用 Rc 和 Weak 解决循环引用
为了处理循环引用问题,Rust 提供了 Rc 和 Weak 两种类型:
Rc<T>: 引用计数类型,允许多个所有者。Weak<T>: 对应于Rc<T>的弱引用版本,不会增加引用计数。
通过使用 Weak 可以打破循环引用,因为 Weak 不会增加其指向的对象的引用计数。
生命周期注解的应用
在 Rust 中,生命周期注解可以帮助编译器更好地理解引用之间的关系。特别是在自引用和循环引用的情况下,生命周期注解尤为重要。
// 定义一个带有生命周期注解的函数
fn process_node<'a>(node: &'a Node) {println!("Processing node with value: {}", node.value);// 访问子节点for child in &node.children {process_node(child); // 递归处理子节点}
}// 使用生命周期注解的结构体方法
impl<'a> Node {fn traverse<'b>(&'a self, visitor: &dyn Fn(&'b Node)) {visitor(self);for child in &self.children {child.traverse(visitor);}}
}
实际代码示例与分析
下面是一个完整的示例,展示了如何创建并操作自引用结构体:
use std::rc::{Rc, Weak};struct Node {value: i32,parent: Option<Weak<Rc<Node>>>,children: Vec<Rc<Node>>,
}impl Node {fn new(value: i32) -> Self {Node {value,parent: None,children: Vec::new(),}}fn add_child(&mut self, child: Rc<Node>) {self.children.push(child.clone());child.parent = Some(Rc::downgrade(&self));}
}fn main() {let root = Rc::new(Node::new(0));let child1 = Rc::new(Node::new(1));let child2 = Rc::new(Node::new(2));root.add_child(child1.clone());root.add_child(child2.clone());println!("Root has {} children", root.children.len());// 访问子节点的父节点if let Some(parent) = child1.parent {if let Some(p) = parent.upgrade() {println!("Child 1's parent is {}", p.value);}}// 遍历树结构root.traverse(&|node| println!("Visiting node with value: {}", node.value));
}
定义 Node 结构:
value: 节点存储的值。parent: 父节点的弱引用,初始为 None。children: 一个向量,存储子节点的强引用。
创建新节点:
new方法初始化一个新的Node实例,此时没有父节点也没有子节点。
添加子节点:
add_child方法接收一个Rc<Node>类型的参数作为子节点。- 将子节点添加到当前节点的
children向量中。 - 更新子节点的
parent字段,使用Rc::downgrade转换为Weak引用。
遍历树结构:
traverse方法使用生命周期注解,递归地遍历整个树结构。
多线程并发
并发与并行概述
- 并发 (Concurrency): 多个任务可以在同一时间间隔内执行,但不一定在同一时刻执行。
- 并行 (Parallelism): 多个任务在同一时刻执行,通常涉及硬件支持。
在 Rust 中,可以通过多线程实现并发,而并行则依赖于多核处理器的支持。
使用多线程
在 Rust 中,可以使用标准库中的 std::thread 模块来创建和管理线程。
创建线程
use std::thread;
use std::time::Duration;fn spawn_thread() {thread::spawn(|| {for i in 1..10 {println!("Thread spawned: {}", i);thread::sleep(Duration::from_millis(1));}});for i in 1..5 {println!("Main thread: {}", i);thread::sleep(Duration::from_millis(1));}
}fn main() {spawn_thread();
}
线程同步:消息传递
在 Rust 中,消息传递是一种常见的线程间通信方式。常用的工具包括 std::sync::mpsc 模块中的通道 (channel)。
使用通道
use std::sync::mpsc;
use std::thread;fn send_messages() {let (tx, rx) = mpsc::channel();thread::spawn(move || {let val = String::from("Hello from the other side!");tx.send(val).unwrap();});let received = rx.recv().unwrap();println!("Got: {}", received);
}fn main() {send_messages();
}
线程同步:锁
Rust 标准库提供了多种锁机制,如 Mutex、RwLock 和 Arc。
使用 Mutex
use std::sync::Mutex;
use std::thread;fn lock_data() {let counter = Mutex::new(0);let mut handles = vec![];for _ in 0..10 {let counter = Mutex::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!("Counter: {}", *counter.lock().unwrap());
}fn main() {lock_data();
}
使用 RwLock
use std::sync::RwLock;
use std::thread;fn read_write_lock() {let data = RwLock::new(String::from("Hello"));let mut handles = vec![];for _ in 0..10 {let data = RwLock::clone(&data);let handle = thread::spawn(move || {let mut d = data.write().unwrap();*d += "!";});handles.push(handle);}for handle in handles {handle.join().unwrap();}println!("Data: {}", *data.read().unwrap());
}fn main() {read_write_lock();
}
线程同步:条件变量和信号量
Rust 标准库提供了 Condvar 和 Semaphore 等高级同步原语。
使用 Condvar
use std::sync::{Arc, Condvar, Mutex};
use std::thread;fn condition_variable() {let pair = Arc::new((Mutex::new(false), Condvar::new()));let pair_clone = Arc::clone(&pair);thread::spawn(move || {let (lock, cvar) = &*pair;let mut started = lock.lock().unwrap();*started = true;cvar.notify_one();});let (lock, cvar) = &*pair;let mut started = lock.lock().unwrap();while !*started {started = cvar.wait(started).unwrap();}println!("Condition variable signaled!");
}fn main() {condition_variable();
}
使用 Semaphore
use std::sync::Semaphore;
use std::thread;fn semaphore_example() {let sem = Semaphore::new(3);let mut handles = vec![];for _ in 0..5 {let sem = sem.clone();let handle = thread::spawn(move || {sem.acquire().unwrap();println!("Acquired semaphore");thread::sleep(std::time::Duration::from_secs(1));sem.release();});handles.push(handle);}for handle in handles {handle.join().unwrap();}
}fn main() {semaphore_example();
}
线程同步:原子操作与内存顺序
Rust 标准库提供了 std::sync::atomic 模块,用于原子操作和内存顺序控制。
原子操作
use std::sync::atomic::{AtomicUsize, Ordering};fn atomic_operations() {let counter = AtomicUsize::new(0);let mut handles = vec![];for _ in 0..10 {let counter = counter.clone();let handle = thread::spawn(move || {counter.fetch_add(1, Ordering::Relaxed);});handles.push(handle);}for handle in handles {handle.join().unwrap();}println!("Counter: {}", counter.load(Ordering::Relaxed));
}fn main() {atomic_operations();
}
内存顺序
use std::sync::atomic::{AtomicUsize, Ordering};fn memory_ordering() {let flag = AtomicUsize::new(0);let data = AtomicUsize::new(0);let mut handles = vec![];let flag_clone = flag.clone();let data_clone = data.clone();let handle1 = thread::spawn(move || {flag_clone.store(1, Ordering::Release);data_clone.store(42, Ordering::Relaxed);});let flag_clone = flag.clone();let data_clone = data.clone();let handle2 = thread::spawn(move || {while flag_clone.load(Ordering::Acquire) == 0 {}assert_eq!(data_clone.load(Ordering::Relaxed), 42);});handles.push(handle1);handles.push(handle2);for handle in handles {handle.join().unwrap();}println!("Memory ordering example completed.");
}fn main() {memory_ordering();
}
基于 Send 和 Sync 的线程安全
在 Rust 中,Send 和 Sync 是两个重要的类型约束,用于确保数据在线程间安全传递。
Send 约束
use std::thread;fn send_constraint() {struct NotSend(u8);impl NotSend {fn new() -> Self {NotSend(0)}}// NotSend 类型不能在线程间传递// let handle = thread::spawn(move || {// println!("NotSend value: {}", NotSend::new().0);// });// 正确的示例let handle = thread::spawn(|| {println!("Send value: {}", 42);});handle.join().unwrap();
}fn main() {send_constraint();
}
Sync 约束
use std::sync::Arc;
use std::thread;fn sync_constraint() {struct NotSync(u8);impl NotSync {fn new() -> Self {NotSync(0)}}// NotSync 类型不能在线程间共享// let shared = NotSync::new();// let handle = thread::spawn(move || {// println!("NotSync value: {}", shared.0);// });// 正确的示例let shared = Arc::new(42);let handle = thread::spawn(move || {println!("Sync value: {}", shared);});handle.join().unwrap();
}fn main() {sync_constraint();
}
文章到此结束,更多相关的信息,请,https://t.me/gtokentool
相关文章:
Rust循环引用与多线程并发
循环引用与自引用 循环引用的概念 循环引用指的是两个或多个对象之间相互持有对方的引用。在 Rust 中,由于所有权和生命周期的严格约束,直接创建循环引用通常会导致编译失败。例如: // 错误的循环引用示例 struct Node {next: Option<B…...
东方隐侠网安瞭望台第8期
谷歌应用商店贷款应用中的 SpyLoan 恶意软件影响 800 万安卓用户 迈克菲实验室的新研究发现,谷歌应用商店中有十多个恶意安卓应用被下载量总计超过 800 万次,这些应用包含名为 SpyLoan 的恶意软件。安全研究员费尔南多・鲁伊斯上周发布的分析报告称&…...
底部导航栏新增功能按键
场景需求: 在底部导航栏添加power案件,单击息屏,长按 关机 如下实现图 借此需求,需要掌握技能: 底部导航栏如何实现新增、修改、删除底部导航栏流程对底部导航栏部分样式如何修改。 比如放不下、顺序排列、坑点如…...
C++ 之弦上舞:string 类与多样字符串操作的优雅旋律
string 类的重要性及与 C 语言字符串对比 在 C 语言中,字符串是以 \0 结尾的字符集合,操作字符串需借助 C 标准库的 str 系列函数,但这些函数与字符串分离,不符合 OOP 思想,且底层空间管理易出错。而在 C 中࿰…...
centos8:Could not resolve host: mirrorlist.centos.org
【1】错误消息: [rootcentos211 redis-7.0.15]# yum update CentOS Stream 8 - AppStream …...
Linux 定时任务 命令解释 定时任务格式详解
目录 时间命令 修改时间和日期 定时任务格式 定时任务执行 查看定时任务进程 重启定时任务 时间命令 #查看时间 [rootlocalhost ~]# date 2021年 07月 23日 星期五 14:38:19 CST --------------------------------------- [rootlocalhost ~]# date %F 2021-07-23 -----…...
aws(学习笔记第十五课) 如何从灾难中恢复(recover)
aws(学习笔记第十五课) 如何从灾难中恢复 学习内容: 使用CloudWatch对服务器进行监视与恢复区域(region),可用区(available zone)和子网(subnet)使用自动扩展(AutoScalingGroup) 1. 使用CloudWatch对服务器进行监视与恢复 整体架构 这里模拟Jenkins Se…...
github webhooks 实现网站自动更新
本文目录 Github Webhooks 介绍Webhooks 工作原理配置与验证应用云服务器通过 Webhook 自动部署网站实现复制私钥编写 webhook 接口Github 仓库配置 webhook以服务的形式运行 app.py Github Webhooks 介绍 Webhooks是GitHub提供的一种通知方式,当GitHub上发生特定事…...
【C语言】递归的内存占用过程
递归 递归是函数调用自身的一种编程技术。在C语言中,递归的实现会占用内存栈(Call Stack),每次递归调用都会在栈上分配一个新的 “栈帧(Stack Frame)”,用于存储本次调用的函数局部变量、返回地…...
365天深度学习训练营-第P6周:VGG-16算法-Pytorch实现人脸识别
🍨 本文为🔗365天深度学习训练营中的学习记录博客🍖 原作者:K同学啊 文为「365天深度学习训练营」内部文章 参考本文所写记录性文章,请在文章开头带上「👉声明」 🍺要求: 保存训练过…...
企业AI助理在数据分析与决策中扮演的角色
在当今这个数据驱动的时代,企业每天都需要处理和分析大量的数据,以支持其业务决策。然而,面对如此庞大的数据量,传统的数据分析方法已经显得力不从心。幸运的是,随着人工智能(AI)技术的不断发展…...
洛谷 B2029:大象喝水 ← 圆柱体体积
【题目来源】https://www.luogu.com.cn/problem/B2029【题目描述】 一只大象口渴了,要喝 20 升水才能解渴,但现在只有一个深 h 厘米,底面半径为 r 厘米的小圆桶 (h 和 r 都是整数)。问大象至少要喝多少桶水才会解渴。 …...
go每日一题:mock打桩、defer、recovery、panic的调用顺序
题目一:单元测试中使用—打桩 打桩概念:使用A替换 原函数B,那么A就是打桩函数打桩原理:运行时,通过一个包,将内存中函数的地址替换为桩函数的地址打桩操作:利用Patch()函…...
STM32F103 HSE时钟倍频以及设置频率函数(新手向,本人也是新手)
HSE_SetSysCLK是野火教程里的,不懂的去这 16-RCC(第3节)使用HSE配置系统时钟并使用MCO输出监控系统时钟_哔哩哔哩_bilibili HSE_AutoSetHSE的算法部分是自己写的,用了一个转接数组。C语言不支持bool所以自己定义了一个boolK代替bool。 AutoHSE.h: /**…...
renderExtraFooter 添加本周,本月,本年
在 Ant Design Vue 中,a-date-picker 组件提供了一个 renderExtraFooter 属性,可以用来渲染额外的页脚内容。你可以利用这个属性来添加“本周”、“本月”和“本年”的按钮。下面是如何在 Vue 2 项目中实现这一功能的具体步骤: 1.确保安装了…...
SprinBoot整合KafKa的使用(详解)
前言 1. 高吞吐量(High Throughput) Kafka 设计的一个核心特性是高吞吐量。它能够每秒处理百万级别的消息,适合需要高频次、低延迟消息传递的场景。即使在大规模分布式环境下,它也能保持很高的吞吐量和性能,支持低延…...
【机器学习】CatBoost 模型实践:回归与分类的全流程解析
一. 引言 本篇博客首发于掘金 https://juejin.cn/post/7441027173430018067。 PS:转载自己的文章也算原创吧。 在机器学习领域,CatBoost 是一款强大的梯度提升框架,特别适合处理带有类别特征的数据。本篇博客以脱敏后的保险数据集为例&#x…...
PyTorch 实现动态输入
使用 PyTorch 实现动态输入:支持训练和推理输入维度不一致的 CNN 和 LSTM/GRU 模型 在深度学习中,处理不同大小的输入数据是一个常见的挑战。许多实际应用需要模型能够灵活地处理可变长度的输入。本文将介绍如何使用 PyTorch 实现支持动态输入的 CNN 和…...
【Linux相关】查看conda路径和conda和cudnn版本、安装cudnn、cuDNN无需登录官方下载链接
【Linux相关】 查看conda路径和conda和cudnn版本 安装cudnn cuDNN无需登录官方下载链接 文章目录 1. 查看信息1.1 查看 Conda 路径1.2 查看 Conda 版本1.3 查看 cuDNN 版本1.4 总结 2. 安装cudnn2.1 安装cudnn步骤2.2 cuDNN无需登录官方下载链接 1. 查看信息 查看Conda 路径、C…...
基于Java Springboot环境保护生活App且微信小程序
一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术:Html、Css、Js、Vue、Element-ui 数据库:MySQL 后端技术:Java、Spring Boot、MyBatis 三、运行环境 开发工具:IDEA/eclipse 微信…...
【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密
在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...
转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...
MVC 数据库
MVC 数据库 引言 在软件开发领域,Model-View-Controller(MVC)是一种流行的软件架构模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种模式有助于提高代码的可维护性和可扩展性。本文将深入探讨MVC架构与数据库之间的关系,以…...
Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器
第一章 引言:语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域,文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量,支撑着搜索引擎、推荐系统、…...
srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
Axios请求超时重发机制
Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式: 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...
智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
Go 并发编程基础:通道(Channel)的使用
在 Go 中,Channel 是 Goroutine 之间通信的核心机制。它提供了一个线程安全的通信方式,用于在多个 Goroutine 之间传递数据,从而实现高效的并发编程。 本章将介绍 Channel 的基本概念、用法、缓冲、关闭机制以及 select 的使用。 一、Channel…...
