测试实例:链表

enum 的一个常见用法就是创建链表(linked-list):

  1. use List::*;
  2. enum List {
  3. // Cons:元组结构体,包含链表的一个元素和一个指向下一节点的指针
  4. Cons(u32, Box<List>),
  5. // Nil:末结点,表明链表结束
  6. Nil,
  7. }
  8. // 可以为 enum 定义方法
  9. impl List {
  10. // 创建一个空的 List 实例
  11. fn new() -> List {
  12. // `Nil` 为 `List` 类型(译注:因 `Nil` 的完整名称是 `List::Nil`)
  13. Nil
  14. }
  15. // 处理一个 List,在其头部插入新元素,并返回该 List
  16. fn prepend(self, elem: u32) -> List {
  17. // `Cons` 同样为 List 类型
  18. Cons(elem, Box::new(self))
  19. }
  20. // 返回 List 的长度
  21. fn len(&self) -> u32 {
  22. // 必须对 `self` 进行匹配(match),因为这个方法的行为取决于 `self` 的
  23. // 取值种类。
  24. // `self` 为 `&List` 类型,`*self` 为 `List` 类型,匹配一个具体的 `T`
  25. // 类型要好过匹配引用 `&T`。
  26. match *self {
  27. // 不能得到 tail 的所有权,因为 `self` 是借用的;
  28. // 因此使用一个对 tail 的引用
  29. Cons(_, ref tail) => 1 + tail.len(),
  30. // (递归的)基准情形(base case):一个长度为 0 的空列表
  31. Nil => 0
  32. }
  33. }
  34. // 返回列表的字符串表示(该字符串是堆分配的)
  35. fn stringify(&self) -> String {
  36. match *self {
  37. Cons(head, ref tail) => {
  38. // `format!` 和 `print!` 类似,但返回的是一个堆分配的字符串,
  39. // 而不是打印结果到控制台上
  40. format!("{}, {}", head, tail.stringify())
  41. },
  42. Nil => {
  43. format!("Nil")
  44. },
  45. }
  46. }
  47. }
  48. fn main() {
  49. // 创建一个空链表
  50. let mut list = List::new();
  51. // 追加一些元素
  52. list = list.prepend(1);
  53. list = list.prepend(2);
  54. list = list.prepend(3);
  55. // 显示链表的最后状态
  56. println!("linked list has length: {}", list.len());
  57. println!("{}", list.stringify());
  58. }

参见:

Box方法