演習: Counter

この演習では、非常に単純なデータ構造を取り上げて、それをジェネリックにします。 これは、 std::collections::HashMap を使って、どの値が現れたかと、それぞれが何回出現したかを追跡します。

Counter の初期バージョンは、u32 値でしか動作しないようにハードコードされています。 追跡する値の型に対して struct とそのメソッドをジェネリックにし、そのようにして Counter が任意の型の値を追跡できるようにしてください。

早く終わったら、 entry メソッドを使って、count メソッドの実装に必要なハッシュ検索の回数を半分にしてみてください。

// Copyright 2023 Google LLC
// SPDX-License-Identifier: Apache-2.0

use std::collections::HashMap;

/// Counter は、型 T の各値が現れた回数を数えます。
struct Counter {
    values: HashMap<u32, u64>,
}

impl Counter {
    /// 新しい Counter を作成します。
    fn new() -> Self {
        Counter {
            values: HashMap::new(),
        }
    }

    /// 指定された値の出現を数えます。
    fn count(&mut self, value: u32) {
        if self.values.contains_key(&value) {
            *self.values.get_mut(&value).unwrap() += 1;
        } else {
            self.values.insert(value, 1);
        }
    }

    /// 指定された値が現れた回数を返します。
    fn times_seen(&self, value: u32) -> u64 {
        self.values.get(&value).copied().unwrap_or_default()
    }
}

fn main() {
    let mut ctr = Counter::new();
    ctr.count(13);
    ctr.count(14);
    ctr.count(16);
    ctr.count(14);
    ctr.count(14);
    ctr.count(11);

    for i in 10..20 {
        println!("saw {} values equal to {}", ctr.times_seen(i), i);
    }

    let mut strctr = Counter::new();
    strctr.count("apple");
    strctr.count("orange");
    strctr.count("apple");
    println!("got {} apples", strctr.times_seen("apple"));
}