以下是我要做的事情:
use std::collections::HashMap;
fn main() {
let mut my_map = HashMap::new();
my_map.insert("a", 1);
my_map.insert("b", 3);
my_map["a"] += 10;
// I expect my_map becomes {"b": 3, "a": 11}
}
但这会引发错误:
Rust 2015
error[E0594]: cannot assign to immutable indexed content
--> src/main.rs:8:5
|
8 | my_map["a"] += 10;
| ^^^^^^^^^^^^^^^^^ cannot borrow as mutable
|
= help: trait `IndexMut` is required to modify indexed content, but it is not implemented for `std::collections::HashMap<&str, i32>`
Rust 2018
error[E0594]: cannot assign to data in a `&` reference
--> src/main.rs:8:5
|
8 | my_map["a"] += 10;
| ^^^^^^^^^^^^^^^^^ cannot assign
我真的不明白这意味着什么,因为我让HashMap
可变。当我尝试更新vector
中的元素时,我得到了预期的结果:
let mut my_vec = vec![1, 2, 3];
my_vec[0] += 10;
println! {"{:?}", my_vec};
// [11, 2, 3]
HashMap
我遇到上述错误有什么不同?有没有办法更新价值?
答案 0 :(得分:59)
可变性索引和可变索引由两个不同的特征提供:Index
和IndexMut
。
目前,HashMap
未实现IndexMut
,Vec
does。
The commit that removed HashMap
's IndexMut
implementation州:
此提交删除了HashMap和BTreeMap上的IndexMut impls 为了能够面向未来的API而不是最终包含一个 IndexSet特征。
我的理解是,假设的IndexSet
特征允许您为HashMap
分配全新的值,而不仅仅是读取或改变现有条目:
let mut map = HashMap::new();
map["key"] = "value";
目前,您可以使用get_mut
:
*my_map.get_mut("a").unwrap() += 10;
或entry
API:
*my_map.entry("a").or_insert(42) += 10;
答案 1 :(得分:2)
我将分享我自己的答案,因为我遇到了这个问题,但是我正在使用Structs,所以在我看来,这种方式有些棘手
use std::collections::HashMap;
#[derive(Debug)]
struct ExampleStruct {
pub field1: usize,
pub field2: f64,
}
fn main() {
let mut example_map = HashMap::new();
&example_map.insert(1usize, ExampleStruct { field1: 50, field2: 184.0});
&example_map.insert(6usize, ExampleStruct { field1: 60, field2: 486.0});
//First Try
(*example_map.get_mut(&1).unwrap()).field1 += 55; //50+55=105
(*example_map.get_mut(&6).unwrap()).field1 -= 25; //60-25=35
//Spliting lines
let op_elem = example_map.get_mut(&6);
let elem = op_elem.unwrap();
(*elem).field2 = 200.0;
let op_ok_elem = example_map.get_mut(&1);
let elem = op_ok_elem.unwrap_or_else(|| panic!("This msg should not appear"));
(*elem).field2 = 777.0;
println!("Map at this point: {:?}", example_map);
let op_err_elem = example_map.get_mut(&8);
let _elem = op_err_elem.unwrap_or_else(|| panic!("Be careful, check you key"));
println!("{:?}", example_map);
}
您可以在document
上玩这个游戏答案 2 :(得分:1)
考虑:
let mut m = std::collections::HashMap::new();
m.insert("a", 1);
m.insert("b", 3);
let k = "c";
如果密钥已经存在:
m.insert(k, 10 + m[k] );
如果密钥不存在:
m.insert(k, 10 + if m.contains_key(k) { m[k] } else { 0 });
m.entry(k).or_insert(0);
m.insert(k, 200 + m[k]);
*m.entry(k).or_insert(0) += 3000;
最后打印值:
println!("{}", m[k]); // 3210
请参阅:
https://doc.rust-lang.org/std/collections/struct.HashMap.html