Skip to content

Commit e800d5a

Browse files
committed
Specify behavior of HashSet::insert
`HashSet::insert` does not replace the value with equal value. Fixes #107581.
1 parent f312650 commit e800d5a

File tree

2 files changed

+23
-1
lines changed

2 files changed

+23
-1
lines changed

library/std/src/collections/hash/set.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -875,7 +875,9 @@ where
875875
/// Returns whether the value was newly inserted. That is:
876876
///
877877
/// - If the set did not previously contain this value, `true` is returned.
878-
/// - If the set already contained this value, `false` is returned.
878+
/// - If the set already contained this value, `false` is returned,
879+
/// and the set is not modified: original value is not replaced,
880+
/// and the value passed as argument is dropped.
879881
///
880882
/// # Examples
881883
///

library/std/src/collections/hash/set/tests.rs

+20
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ use super::HashSet;
33

44
use crate::panic::{catch_unwind, AssertUnwindSafe};
55
use crate::sync::atomic::{AtomicU32, Ordering};
6+
use crate::sync::Arc;
67

78
#[test]
89
fn test_zero_capacities() {
@@ -502,3 +503,22 @@ fn const_with_hasher() {
502503
const X: HashSet<(), ()> = HashSet::with_hasher(());
503504
assert_eq!(X.len(), 0);
504505
}
506+
507+
#[test]
508+
fn test_insert_does_not_overwrite_the_value() {
509+
let first_value = Arc::new(17);
510+
let second_value = Arc::new(17);
511+
512+
let mut set = HashSet::new();
513+
let inserted = set.insert(first_value.clone());
514+
assert!(inserted);
515+
516+
let inserted = set.insert(second_value);
517+
assert!(!inserted);
518+
519+
assert!(
520+
Arc::ptr_eq(set.iter().next().unwrap(), &first_value),
521+
"Insert must not overwrite the value, so the contained value pointer \
522+
must be the same as first value pointer we inserted"
523+
);
524+
}

0 commit comments

Comments
 (0)