#variables #scope #hash-set #hash-map #scoping

quickscope

Multi-layer HashMap and HashSet implementations for performant representation of variable scopes

8 releases

0.2.0 Jan 28, 2022
0.1.6 Jan 11, 2021
0.1.5 Oct 11, 2020
0.1.3 Sep 22, 2020

#1016 in Data structures

Download history 44/week @ 2024-09-13 121/week @ 2024-09-20 63/week @ 2024-09-27 47/week @ 2024-10-04 84/week @ 2024-10-11 104/week @ 2024-10-18 125/week @ 2024-10-25 194/week @ 2024-11-01 74/week @ 2024-11-08 61/week @ 2024-11-15 59/week @ 2024-11-22 65/week @ 2024-11-29 153/week @ 2024-12-06 111/week @ 2024-12-13 63/week @ 2024-12-20 131/week @ 2024-12-27

481 downloads per month
Used in 8 crates (3 directly)

MIT license

44KB
1K SLoC

quickscope

Crates.io Docs.rs

Multi-layer HashMap and HashSet implementations for performant representation of variable scopes.

Installation

Please use cargo-edit to always add the latest version of this library:

cargo add quickscope

Purpose

This crate contains two data structures, ScopeMap and ScopeSet, for representing variable scopes and, in the case of ScopeMap, associated variable values. Access operations are such that variables in higher (i.e. more specific) scopes override variables in lower (i.e. more general) scopes.

Rationale (i.e. "why another one of these?")

I know I'm not the first one to do this! There are other, equally usable crates for the same purpose (see: hash-chain, chainmap, chain-map), but their implementations are such that lookups are O(n) worst-case in relation to layer count. I found that this didn't suit my needs.

This crate is optimized so that lookups are O(1) on average in relation to layer count. The trade-off is that popping layers is an O(n) operation in relation to the number of keys stored in the removed layer. I think this is an acceptable compromise in use cases requiring fast lookups across a large number of layers.

Example

let mut vars = ScopeMap::new();

// Define two variables in main scope
vars.define("a", 1);
vars.define("b", 2);

// Add a child scope
vars.push_layer();

// Override value of `a`
vars.define("a", 3);

assert_eq!(Some(&3), vars.get("a"));
assert_eq!(Some(&2), vars.get("b"));

// Remove child scope
vars.pop_layer();

// Value of `a` is no longer overridden
assert_eq!(Some(&1), vars.get("a"));

Why is it missing X feature from the regular HashMap/HashSet?

ScopeMap and ScopeSet are optimized for representing variable scopes. As such, they are missing many of the typical methods found in more general-purpose HashMap/HashSet implementations. If there's a feature that you feel should be added, feel free to submit a PR or post an issue about it.

License

This library is distributed under the MIT License. See LICENSE for details.

Dependencies

~1MB
~18K SLoC