summaryrefslogtreecommitdiffstats
path: root/vendor/elsa/examples/string_interner.rs
blob: fd039f7ba5e8117ce3254d31051e3193abd120f4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
use std::collections::BTreeSet;
use std::convert::AsRef;

use elsa::FrozenIndexSet;

struct StringInterner {
    set: FrozenIndexSet<String>,
}

impl StringInterner {
    fn new() -> Self {
        StringInterner {
            set: FrozenIndexSet::new(),
        }
    }

    fn get_or_intern<T>(&self, value: T) -> usize
    where
        T: AsRef<str>,
    {
        // TODO use Entry in case the standard Entry API gets improved
        // (here to avoid premature allocation or double lookup)
        self.set.insert_full(value.as_ref().to_string()).0
    }

    fn get<T>(&self, value: T) -> Option<usize>
    where
        T: AsRef<str>,
    {
        self.set.get_full(value.as_ref()).map(|(i, _r)| i)
    }

    fn resolve(&self, index: usize) -> Option<&str> {
        self.set.get_index(index)
    }
}

fn main() {
    let interner = StringInterner::new();
    let lonely = interner.get_or_intern("lonely");
    let best_friend = interner.get_or_intern("best friend");
    let threes_a_crowd = interner.get_or_intern("threes a crowd");
    let rando = interner.get_or_intern("rando");
    let _facebook = interner.get_or_intern("facebook");

    let best_friend_2 = interner.get_or_intern("best friend");
    let best_friend_3 = interner.get("best friend").unwrap();

    let best_friend_ref = interner.resolve(best_friend).unwrap();

    let mut set = BTreeSet::new();
    set.insert(lonely);
    set.insert(best_friend);
    set.insert(threes_a_crowd);
    set.insert(rando);
    set.insert(best_friend_2);
    assert_eq!(set.len(), 4);
    assert_eq!(best_friend, best_friend_2);
    assert_eq!(best_friend_2, best_friend_3);
    assert_eq!(best_friend_ref, "best friend");
}