ํ•ด๋‹ต

use std::collections::HashMap;
use std::hash::Hash;

/// Counter๋Š” ๊ฐ T ํƒ€์ž… ๊ฐ’์ด ํ‘œ์‹œ๋œ ํšŸ์ˆ˜๋ฅผ ๊ณ„์‚ฐํ•ฉ๋‹ˆ๋‹ค.
struct Counter<T: Eq + Hash> {
    values: HashMap<T, u64>,
}

impl<T: Eq + Hash> Counter<T> {
    /// ์ƒˆ Counter๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.
    fn new() -> Self {
        Counter { values: HashMap::new() }
    }

    /// ์ง€์ •๋œ ๊ฐ’์˜ ๋ฐœ์ƒ ํšŸ์ˆ˜๋ฅผ ์…‰๋‹ˆ๋‹ค.
    fn count(&mut self, value: T) {
        *self.values.entry(value).or_default() += 1;
    }

    /// ์ง€์ •๋œ ๊ฐ’์ด ํ‘œ์‹œ๋œ ํšŸ์ˆ˜๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
    fn times_seen(&self, value: T) -> 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!("{} ๊ฐœ์˜ {} ๊ฐ’์„ ๋ฐœ๊ฒฌํ–ˆ์Šต๋‹ˆ๋‹ค.", ctr.times_seen(i), i);
    }

    let mut strctr = Counter::new();
    strctr.count("์‚ฌ๊ณผ");
    strctr.count("์˜ค๋ Œ์ง€");
    strctr.count("์‚ฌ๊ณผ");
    println!("์‚ฌ๊ณผ {}๊ฐœ ๋ฐ›์Œ", strctr.times_seen("์‚ฌ๊ณผ"));
}