Skip to content

Commit

Permalink
Merge pull request #290 from bkragl/entry_at
Browse files Browse the repository at this point in the history
Add method to get a map entry by index
  • Loading branch information
cuviper authored Jan 12, 2024
2 parents 7e0862c + 5fc8277 commit 84f772d
Show file tree
Hide file tree
Showing 3 changed files with 149 additions and 1 deletion.
14 changes: 13 additions & 1 deletion src/map.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ use alloc::vec::Vec;
#[cfg(feature = "std")]
use std::collections::hash_map::RandomState;

use self::core::IndexMapCore;
use self::core::{IndexMapCore, IndexedEntry};
use crate::util::{third, try_simplify_range};
use crate::{Bucket, Entries, Equivalent, HashValue, TryReserveError};

Expand Down Expand Up @@ -899,6 +899,18 @@ impl<K, V, S> IndexMap<K, V, S> {
self.as_entries_mut().get_mut(index).map(Bucket::ref_mut)
}

/// Get an entry in the map by index for in-place manipulation.
///
/// Valid indices are *0 <= index < self.len()*
///
/// Computes in **O(1)** time.
pub fn get_index_entry(&mut self, index: usize) -> Option<IndexedEntry<'_, K, V>> {
if index >= self.len() {
return None;
}
Some(IndexedEntry::new(&mut self.core, index))
}

/// Returns a slice of key-value pairs in the given range of indices.
///
/// Valid indices are *0 <= index < self.len()*
Expand Down
106 changes: 106 additions & 0 deletions src/map/core.rs
Original file line number Diff line number Diff line change
Expand Up @@ -754,9 +754,115 @@ impl<K: fmt::Debug, V> fmt::Debug for VacantEntry<'_, K, V> {
}
}

/// A view into an occupied entry in a `IndexMap` obtained by index.
///
/// This struct is constructed from the `get_index_entry` method on `IndexMap`.
pub struct IndexedEntry<'a, K, V> {
map: &'a mut IndexMapCore<K, V>,
// We have a mutable reference to the map, which keeps the index
// valid and pointing to the correct entry.
index: usize,
}

impl<'a, K, V> IndexedEntry<'a, K, V> {
pub(crate) fn new(map: &'a mut IndexMapCore<K, V>, index: usize) -> Self {
Self { map, index }
}

/// Gets a reference to the entry's key in the map.
pub fn key(&self) -> &K {
&self.map.entries[self.index].key
}

/// Gets a reference to the entry's value in the map.
pub fn get(&self) -> &V {
&self.map.entries[self.index].value
}

/// Gets a mutable reference to the entry's value in the map.
///
/// If you need a reference which may outlive the destruction of the
/// `IndexedEntry` value, see `into_mut`.
pub fn get_mut(&mut self) -> &mut V {
&mut self.map.entries[self.index].value
}

/// Sets the value of the entry to `value`, and returns the entry's old value.
pub fn insert(&mut self, value: V) -> V {
mem::replace(self.get_mut(), value)
}

/// Return the index of the key-value pair
#[inline]
pub fn index(&self) -> usize {
self.index
}

/// Converts into a mutable reference to the entry's value in the map,
/// with a lifetime bound to the map itself.
pub fn into_mut(self) -> &'a mut V {
&mut self.map.entries[self.index].value
}

/// Remove and return the key, value pair stored in the map for this entry
///
/// Like `Vec::swap_remove`, the pair is removed by swapping it with the
/// last element of the map and popping it off. **This perturbs
/// the position of what used to be the last element!**
///
/// Computes in **O(1)** time (average).
pub fn swap_remove_entry(self) -> (K, V) {
self.map.swap_remove_index(self.index).unwrap()
}

/// Remove and return the key, value pair stored in the map for this entry
///
/// Like `Vec::remove`, the pair is removed by shifting all of the
/// elements that follow it, preserving their relative order.
/// **This perturbs the index of all of those elements!**
///
/// Computes in **O(n)** time (average).
pub fn shift_remove_entry(self) -> (K, V) {
self.map.shift_remove_index(self.index).unwrap()
}

/// Remove the key, value pair stored in the map for this entry, and return the value.
///
/// Like `Vec::swap_remove`, the pair is removed by swapping it with the
/// last element of the map and popping it off. **This perturbs
/// the position of what used to be the last element!**
///
/// Computes in **O(1)** time (average).
pub fn swap_remove(self) -> V {
self.swap_remove_entry().1
}

/// Remove the key, value pair stored in the map for this entry, and return the value.
///
/// Like `Vec::remove`, the pair is removed by shifting all of the
/// elements that follow it, preserving their relative order.
/// **This perturbs the index of all of those elements!**
///
/// Computes in **O(n)** time (average).
pub fn shift_remove(self) -> V {
self.shift_remove_entry().1
}
}

impl<K: fmt::Debug, V: fmt::Debug> fmt::Debug for IndexedEntry<'_, K, V> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct(stringify!(IndexedEntry))
.field("index", &self.index)
.field("key", self.key())
.field("value", self.get())
.finish()
}
}

#[test]
fn assert_send_sync() {
fn assert_send_sync<T: Send + Sync>() {}
assert_send_sync::<IndexMapCore<i32, i32>>();
assert_send_sync::<Entry<'_, i32, i32>>();
assert_send_sync::<IndexedEntry<'_, i32, i32>>();
}
30 changes: 30 additions & 0 deletions src/map/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -359,6 +359,36 @@ fn occupied_entry_key() {
}
}

#[test]
fn get_index_entry() {
let mut map = IndexMap::new();

assert!(map.get_index_entry(0).is_none());

map.insert(0, "0");
map.insert(1, "1");
map.insert(2, "2");
map.insert(3, "3");

assert!(map.get_index_entry(4).is_none());

{
let e = map.get_index_entry(1).unwrap();
assert_eq!(*e.key(), 1);
assert_eq!(*e.get(), "1");
assert_eq!(e.swap_remove(), "1");
}

{
let mut e = map.get_index_entry(1).unwrap();
assert_eq!(*e.key(), 3);
assert_eq!(*e.get(), "3");
assert_eq!(e.insert("4"), "3");
}

assert_eq!(*map.get(&3).unwrap(), "4");
}

#[test]
fn keys() {
let vec = vec![(1, 'a'), (2, 'b'), (3, 'c')];
Expand Down

0 comments on commit 84f772d

Please sign in to comment.