Rust多线程之数据共享

   2023-02-09 学习力0
核心提示:1. 怎么创建线程我们都知道Java中通过实现Runnable接口或继承Thread类,来完成一个线程的创建,那么在Rust中是如何实现一个线程的呢?代码如下。fn how_to_create_a_thread(){// 创建一个线程let thread_handle = thread::spawn(|| {println!("Thread inner")

1. 怎么创建线程

我们都知道Java中通过实现Runnable接口或继承Thread类,来完成一个线程的创建,那么在Rust中是如何实现一个线程的呢?代码如下。

fn how_to_create_a_thread(){
	// 创建一个线程
    let thread_handle = thread::spawn(|| {
        println!("Thread inner");
    });
    // 阻塞线程,并等待其自己执行完毕
    thread_handle.join().unwrap();
}

在Rust中,在std::thread,可以直接通过thread::spawn(||{})方式创建出一个线程,并且返回该线程JoinHandle,可以通过JoinHandle进行join操作。

2. 如何实现线程数据共享?

2.1 单线程

不难猜出,同Java类似,通过加锁的方式保证其数据安全。我们来看具体实现代码。

fn test_single_thread(){
    let m = Mutex::new(5);
    {
        let  mut num = m.lock().unwrap();
        *num = 6;
    }
    println!("m = {:?}", m);
}

在Rust中,通过std::sync::Mutex类,Mutex::new(t: T),将需要共享的数据放进去即可,通过,lock()方法,对数据加锁并获取数据,这样就可以对加锁的数据进行操作。 但是与Java有所不同的是:Rust中的锁实现了Drop接口,会自动释放,不需要手动Unlock。用Java代码解释,就是类似实现了AutoCloseable接口,可以实现自动关闭。代码如下:

public class Mutex implement AutoCloseable{
    public void close() throws Exception{
    }
}
try(Mutex mut = new Mutex(5)){
    
}
2.2 多线程

单线程实现数据共享看起来蛮简单的,直接创建一个线程,然后运行就完了,就看不出来。那如何实现多线程呢?代码如下:

// 多线程和锁
fn test_mulit_thread() {
	// 使用Arc实现clone功能
    // 对Mutex::new生成的对象实现clone功能
    // 如果不实现clone,只能移动一次
    let lock_sub = Arc::new(Mutex::new(0));

    let mut thread_list = vec![];

    for _ in 0..10 {
        // clone 锁对象,否则下方 move一次后,其他for循环将获取不到锁对象
        let lock = Arc::clone(&lock_sub);
        
        let thandle = thread::spawn(move || {
            let mut num = lock.lock().unwrap();
            *num += 1
        });
        thread_list.push(thandle);
    }

    for handle in thread_list {
        handle.join().unwrap();
    }
    println!("Result  : {}", *lock_sub.lock().unwrap());

上方例子是标准实现多线程数据共享的方式。其实有如下疑问,可以说一下。

  1. 为什么要使用Arc对象包装?

    Arc原名:atomically reference counted,原子引用计数。是Rc类型的原子扩展。包含clone方法,对对象进行clone。便于多线程操作同一个对象。

  2. 线程中move问题

    move可以从字面意思理解,就是移动,把lock对象从一个作用域移动到了线程类。Rust对对象的所有权有严格控制,这个需要了解。

2.2.1 疑问

通过上面的方式实现了多线程,其实还有蛮多疑问的, 如果我就是不用Arc对象,而是采用普通对象,或者说Rc对象了,就真的不能实现多线程间的数据共享吗?

2.2.1.1 普通对象

普通对象即:直接一个对象。 这有的方式不行,因为多线程间共享变量就必须用到多个对象,即对象的多副本,那么就要实现Rc对象。

2.2.1.2 用Rc对象实现多数据共享

通过Rc对象,来实现多线程数据共享,代码如下。

fn test_mulit_thread_by_rc() {

    let lock_sub = Rc::new(Mutex::new(0));

    let mut thread_list = vec![];

    for _ in 0..10 {
        let lock = Rc::clone(&lock_sub);
        
        let thandle = thread::spawn(move || {
            let mut num = lock.lock().unwrap();
            *num += 1
        });
        thread_list.push(thandle);
    }

    for handle in thread_list {
        handle.join().unwrap();
    }
    println!("Result  : {}", *lock_sub.lock().unwrap());
}

代码同Arc没什么不同,就是把Arc对象换成Rc对象,然后试着编译。出现如下错误:

45  |           let thandle = thread::spawn(move || {
    |  _______________________^^^^^^^^^^^^^_-
    | |                       |
    | |                       `std::rc::Rc<std::sync::Mutex<i32>>` cannot be sent between threads safely

标红的提示:``std::rc::Rc<std::sync::Mutex> cannot be sent between threads safely,就是说Rc不是线程安全的。没办法,编译不过去。不死心,我又换了一种写法。代码如下。

fn test_mulit_thread_by_rc() {
    let lock_sub = Rc::new(0);
    let mut thread_list = vec![];
    for _ in 0..10 {
        let mut lock = Rc::clone(&lock_sub);
        
        let thandle = thread::spawn(move || {
            *lock += 1
        });
        thread_list.push(thandle);
    }
    for handle in thread_list {
        handle.join().unwrap();
    }
    println!("Result  : {}", *lock_sub);
}

我放弃锁了,不加锁,能行吗? 通过编入后,还是出现了 ``std::rc::Rc<std::sync::Mutex<i32>> cannot be sent between threads safely`这个错误,看来和Mutex对象没有关系啦。

3. 总结

本文主要讲述了如何实现线程,多线程,以及线程间的数据共享问题。但是我们知道,在多线程中,多线程操作数据,会出现死锁等问题,这个还没有说道。以后在慢慢聊。希望本文对大家有所帮助。谢谢!

 
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • bloom-server 基于 rust 编写的 rest api cache 中间件
    bloom-server 基于 rust 编写的 rest api cache
    bloom-server 基于 rust 编写的 rest api cache 中间件,他位于lb 与api worker 之间,使用redis 作为缓存内容存储, 我们需要做的就是配置proxy,同时他使用基于share 的概念,进行cache 的分布存储,包含了请求端口(proxy,访问数据) 以及cache 控制端口(
    03-08
  • #新闻拍一拍# Oracle 调研如何避免让 Java 开发者投奔 Rust 和 Kotlin | Linux 中国
    #新闻拍一拍# Oracle 调研如何避免让 Java 开发
     导读:• 英特尔对迟迟不被 Linux 主线接受的 SGX Enclave 进行了第 38 次修订 • ARM 支持开源的 Panfrost Gallium3D 驱动本文字数:977,阅读时长大约:1分钟作者:硬核老王Oracle 调研如何避免让 Java 开发者投奔 Rust 和 KotlinOracle 委托分析公司 Omd
    03-08
  • Linux系统下Rust快速安装:国内镜像加速
    Linux系统下Rust快速安装:国内镜像加速
    官方网址和方法Install Rust - Rust Programming Language然而速度慢得让人难以置信。利用国内镜像进行windows的Linux子系统的Rust安装。rust 使用国内镜像,快速安装方法参考:RUST安装慢怎么办,使用镜像方式安装_网络_为中华之崛起而编程-CSDN博客我的操作
    03-08
  • Rust到底值不值得学--Rust对比、特色和理念
    前言其实我一直弄不明白一点,那就是计算机技术的发展,是让这个世界变得简单了,还是变得更复杂了。当然这只是一个玩笑,可别把这个问题当真。然而对于IT从业者来说,这可不是一个玩笑。几乎每一次的技术发展,都让这个生态变得更为复杂。“英年早秃”已经成
    03-08
  • 超33000行新代码,为Linux内核添加Rust支持的补丁已准备就绪
    超33000行新代码,为Linux内核添加Rust支持的补
    https://mp.weixin.qq.com/s/oKw9aBJSdmRoO6-rbLAkNw7 月 4 日,一套修订后的补丁被提交至 Linux 内核的邮件列表中,该补丁为在 Linux 内核中以 Rust 作为辅助编程语言提供了支持,借助 Rust 可以提高 Linux 内核和内存的安全。整套补丁包含 17 个子项,不光
    03-08
  • 【译】Rust 的 Result 类型入门
    【译】Rust 的 Result 类型入门
    A Primer on Rust’s Result Type 译文原文链接:https://medium.com/@JoeKreydt/a-primer-on-rusts-result-type-66363cf18e6a原文作者:Joe Kreydt译文出处:https://github.com/suhanyujie/article-transfer-rs译者:suhanyujietips:水平有限,翻译不当之
    03-08
  • Rust实战系列-基本语法
    Rust实战系列-基本语法
    主要介绍 Rust 的语法、基本类型和数据结构,通过实现一个简单版 grep 命令行工具,来理解 Rust 独有的特性。本文是《Rust in action》学习总结系列的第二部分,更多内容请看已发布文章:一、Rust实战系列-Rust介绍“主要介绍 Rust 的语法、基本类型和数据结
    03-08
  • 全栈程序员的新玩具Rust(三)板条箱
    上次用到了stdout,这次我们来写一个更复杂一点的游戏rust的标准库叫做std,默认就会引入。这次我们要用到一个随机数函数,而随机数比较尴尬的一点是这玩意不在标准库中,我们要额外依赖一个库。很多编程方案都有自己的模块化库系统,rust也不例外,不过rust
    02-10
  • 全栈程序员的新玩具Rust(六)第一个WASM程序
    全栈程序员的新玩具Rust(六)第一个WASM程序
    先上代码https://gitee.com/lightsever/rust_study/tree/master/wasm_hello01webassembly就不用再赘述了,耳朵里面快磨出茧子来了。rustwasm是火狐自家的玩具,让我们来继续做实验,让rust飞起来吧。环境安装安装好rust环境之后仍然需要 一个 wasm 工具包carg
    02-10
  • 【Rust】标准库-Result rust数据库
    环境Rust 1.56.1VSCode 1.61.2概念参考:https://doc.rust-lang.org/stable/rust-by-example/std/result.html示例main.rsmod checked {#[derive(Debug)]pub enum MathError {DivisionByZero,NonPositiveLogarithm,NegativeSquareRoot,}pub type MathResult =
    02-09
点击排行