1 unstable release

0.1.0 Aug 20, 2022

#2312 in Rust patterns

Download history 72/week @ 2024-07-15 84/week @ 2024-07-22 69/week @ 2024-07-29 81/week @ 2024-08-05 106/week @ 2024-08-12 128/week @ 2024-08-19 166/week @ 2024-08-26 35/week @ 2024-09-02 42/week @ 2024-09-09 33/week @ 2024-09-16 43/week @ 2024-09-23 73/week @ 2024-09-30 97/week @ 2024-10-07 22/week @ 2024-10-14 27/week @ 2024-10-21 35/week @ 2024-10-28

185 downloads per month

MIT/Apache

14KB
163 lines

omniswap: a crate to swap values between possibly-overlapping references

Motivating Example

You cannot simply use std::mem::swap to replace values within an array:

let mut a = [1, 2, 3];
// You cannot prove their disjointness!
std::mem::swap(&mut a[0], &mut a[2]);

You get the following message:

error[E0499]: cannot borrow `a[_]` as mutable more than once at a time
 --> src/main.rs:4:31
  |
4 |     std::mem::swap(&mut a[0], &mut a[2]);
  |     -------------- ---------  ^^^^^^^^^ second mutable borrow occurs here
  |     |              |
  |     |              first mutable borrow occurs here
  |     first borrow later used by call
  |
  = help: consider using `.split_at_mut(position)` or similar method to obtain two mutable non-overlapping sub-slices

You can use the dedicated <[T]>::swap instead:

let mut a = [1, 2, 3];
a.swap(0, 2);

But how about two-dimensional arrays?

let mut a = [[1, 2], [3, 4]];
// You cannot prove their disjointness!
std::mem::swap(&mut a[0][0], &mut a[1][1]);

This is not as simple as the first one.

Solution

This crate solves the problem by providing a generic framework for sentinel-based swapping.

The idea is simple: it leaves a dummy value behind to safely move values around:

let mut a = [[1, 2], [3, 4]];
let tmp = std::mem::replace(&mut a[0][0], 0);
let tmp = std::mem::replace(&mut a[1][1], tmp);
a[0][0] = tmp;
# assert_eq!(a, [[4, 2], [3, 1]]);

However, in Rust, the best sentinel value differs between types.

The macro swap! automatically chooses the best sentinel and provides the same interface as std::mem::swap:

let mut a = [[1, 2], [3, 4]];
omniswap::swap!(&mut a[0][0], &mut a[1][1]);
# assert_eq!(a, [[4, 2], [3, 1]]);

Usage

Simply use swap! where you want to use std::mem::swap:

let mut x = 42;
let mut y = 84;
omniswap::swap!(&mut x, &mut y);

See swap! for detailed usages.

Other APIs

The crate provides the following variants:

  • rotate! -- swaps more than two values at once

The crate also exposes take! and Replace. These are primitives used in swap! and rotate!.

No runtime deps