更改或自定义关键字类型

任何实现了 EqHash trait 的类型都可以充当 HashMap 的键。这包括:

  • bool (当然这个用处不大,因为只有两个可能的键)
  • intunit,以及其他整数类型
  • String&str(友情提示:如果使用 String 作为键来创建 HashMap,则可以 将 &str 作为散列表的 .get() 方法的参数,以获取值)

注意到 f32f64 没有实现 Hash,这很大程度上是由于若使用浮点数作为 散列表的键,浮点精度误差会很容易导致错误。

对于所有的集合类(collection class),如果它们包含的类型都分别实现了 EqHash,那么这些集合类也就实现了 EqHash。例如,若 T 实现了 Hash,则 Vec<T> 也实现了 Hash

对自定义类型可以轻松地实现 EqHash,只需加上一行代码:#[derive(PartialEq, Eq, Hash)]

编译器将会完成余下的工作。如果你想控制更多的细节,你可以手动 实现 Eq 和/或 Hash。本指南不包含实现 Hash 的细节内容。

为了试验 HashMap 中的 struct,让我们试着做一个非常简易的用户登录系统:

  1. use std::collections::HashMap;
  2. // Eq 要求你对此类型推导 PartiaEq。
  3. #[derive(PartialEq, Eq, Hash)]
  4. struct Account<'a>{
  5. username: &'a str,
  6. password: &'a str,
  7. }
  8. struct AccountInfo<'a>{
  9. name: &'a str,
  10. email: &'a str,
  11. }
  12. type Accounts<'a> = HashMap<Account<'a>, AccountInfo<'a>>;
  13. fn try_logon<'a>(accounts: &Accounts<'a>,
  14. username: &'a str, password: &'a str){
  15. println!("Username: {}", username);
  16. println!("Password: {}", password);
  17. println!("Attempting logon...");
  18. let logon = Account {
  19. username: username,
  20. password: password,
  21. };
  22. match accounts.get(&logon) {
  23. Some(account_info) => {
  24. println!("Successful logon!");
  25. println!("Name: {}", account_info.name);
  26. println!("Email: {}", account_info.email);
  27. },
  28. _ => println!("Login failed!"),
  29. }
  30. }
  31. fn main(){
  32. let mut accounts: Accounts = HashMap::new();
  33. let account = Account {
  34. username: "j.everyman",
  35. password: "password123",
  36. };
  37. let account_info = AccountInfo {
  38. name: "John Everyman",
  39. email: "j.everyman@email.com",
  40. };
  41. accounts.insert(account, account_info);
  42. try_logon(&accounts, "j.everyman", "psasword123");
  43. try_logon(&accounts, "j.everyman", "password123");
  44. }