#trie #prefix-tree #digital-tree #retrieval-tree

plain_trie

Classic trie implementation capable of mapping any T to char iterator

12 stable releases

new 6.0.0 Jan 5, 2025
5.0.2 Dec 15, 2024
4.0.1 Nov 28, 2024
3.0.0 Nov 25, 2024
1.0.2 Jul 22, 2024

#430 in Data structures

Download history 13/week @ 2024-09-11 10/week @ 2024-09-18 9/week @ 2024-09-25 2/week @ 2024-10-02 243/week @ 2024-10-16 59/week @ 2024-10-23 89/week @ 2024-11-13 280/week @ 2024-11-20 333/week @ 2024-11-27 305/week @ 2024-12-04 124/week @ 2024-12-11 15/week @ 2024-12-18

450 downloads per month

MIT license

56KB
1.5K SLoC

(plain) trie

  • plain trie is classic retrieval tree implementation with fixed size alphabet per node
  • allows to use any impl Iterator<Item = char> type as key
  • oob support for English small letters only
  • all methods with classic trie asymptotic computational complexity
  • customizable alphabet support

basic usage

let mut trie = Trie::new();
let key = || "oomph".chars();
let val = 333;

_ = trie.ins(key(), val);
match trie.acq(key()) {
    AcqRes::Ok(v) => assert_eq!(&val, v),
    _ => panic!("Expected AcqRes::Ok(_).")
}

let val = 444;
_ = trie.ins(key(), val);
match trie.acq(key()) {
    AcqRes::Ok(v) => assert_eq!(&val, v),
    _ => panic!("Expected AcqRes::Ok(_).")
}

let catch = catch_unwind(move|| _ = trie.ins("A".chars(), 0));
assert!(catch.is_err());

custom alphabet implementation

  • use Trie::new_with in conjunction with implementation for char-index conversion function and apposite alphabet length
  • example bellow shows sample implementation for alphabet extended with capital letters
use plain_trie::{AcqRes, RemRes, Trie};

const ALPHABET_LEN: u32 = 52;

fn ix(c: char) -> usize {
    let big_a = u32::from('A');
    let big_z = u32::from('Z');

    let sma_a = u32::from('a');
    let sma_z = u32::from('z');

    let cod_poi = u32::from(c);
    (match cod_poi {
        c if c >= big_a && c <= big_z => c - big_a,
        c if c >= sma_a && c <= sma_z => c - sma_a + ALPHABET_LEN / 2,
        _ => {
            panic!("Index conversion impossible.")
        }
    }) as usize
}

#[test]
fn test() {
    let mut trie = Trie::new_with(ix, None, ALPHABET_LEN as usize);

    let kv_1 = ("AZ", 1);
    let kv_2 = ("az", 2);

    for kv in [kv_1, kv_2] {
        let res = trie.ins(kv.0.chars(), kv.1);
        assert_eq!(true, res.is_ok());
    }

    for kv in [kv_1, kv_2] {
        let res = trie.acq(kv.0.chars());
        assert_eq!(AcqRes::Ok(&kv.1), res);
    }

    for kv in [kv_1, kv_2] {
        let res = trie.rem(kv.0.chars());
        assert_eq!(RemRes::Ok(kv.1), res);
    }
}

No runtime deps

Features