MASSLESS LTD.

Concurrency and Parallelism

15 February 2025

Concurrency and Parallelism

Rust provides powerful tools for concurrent and parallel programming, all while ensuring memory safety and preventing data races. In this article, we'll explore several approaches to achieve concurrency and parallelism in Rust, including thread spawning, message passing with channels, and leveraging libraries like rayon for data parallelism.


Concurrency in Rust

Threads in Rust

Rust's standard library offers built-in support for threads through the std::thread module. Each thread has its own stack and can run concurrently with others.

Example: Spawning a Thread

use std::thread;
use std::time::Duration;

fn main() {
    let handle = thread::spawn(|| {
        for i in 1..5 {
            println!("Spawned thread says: {}", i);
            thread::sleep(Duration::from_millis(500));
        }
    });

    // The main thread continues execution concurrently
    for i in 1..3 {
        println!("Main thread says: {}", i);
        thread::sleep(Duration::from_millis(700));
    }

    // Wait for the spawned thread to finish
    handle.join().unwrap();
}

In this example, a new thread is spawned to execute a block of code while the main thread runs concurrently.


Message Passing with Channels

For inter-thread communication, Rust provides channels via the std::sync::mpsc module. Channels allow threads to safely exchange data without sharing mutable state.

Example: Using Channels for Communication

use std::sync::mpsc;
use std::thread;
use std::time::Duration;

fn main() {
    // Create a channel
    let (tx, rx) = mpsc::channel();

    thread::spawn(move || {
        let messages = vec!["Hello", "from", "Rust"];
        for msg in messages {
            tx.send(msg).unwrap();
            thread::sleep(Duration::from_millis(500));
        }
    });

    // Receive messages from the channel
    for received in rx {
        println!("Received: {}", received);
    }
}

Channels provide a safe and straightforward way to share data between threads without risking data races.


Parallelism with Rust

Data Parallelism with Rayon

For tasks that can be parallelised over collections, the rayon library is a great choice. It allows you to easily convert iterators into parallel iterators, harnessing the power of multiple cores.

Example: Parallel Iteration with Rayon

use rayon::prelude::*;

fn main() {
    let numbers: Vec<i32> = (1..=10).collect();
    let squared_numbers: Vec<i32> = numbers.par_iter()
        .map(|&x| x * x)
        .collect();

    println!("Squared numbers: {:?}", squared_numbers);
}

With Rayon, you can parallelise computations over large datasets with minimal code changes, improving performance with minimal overhead.


Best Practices for Safe Concurrency

  • Avoid Shared Mutable State: Leverage Rust's ownership and borrowing rules to prevent data races.
  • Use Channels for Communication: When threads need to communicate, channels provide a safe alternative to shared memory.
  • Leverage Concurrency Libraries: Use libraries like Rayon for parallelising data-intensive tasks efficiently.
  • Mind Blocking Operations: Ensure long-running or blocking operations do not stall other threads unnecessarily.

Conclusion

Rust's approach to concurrency and parallelism emphasises safety and performance. Whether you're spawning threads, using channels for inter-thread communication, or parallelising computations with Rayon, Rust provides the tools needed to build robust and efficient concurrent programs. The language's compile-time guarantees help catch common pitfalls, ensuring that your concurrent code remains safe and reliable.

Happy coding, and enjoy exploring Rust's concurrency and parallelism features!

Rust for beginner's guide overview

Go to previous page: Functional Programming

Go to next page: Iterators and Closures