The wavelet tree [1] is a compact data structure that for a sequence $S$ of length $n$ over an alphabet of size $\sigma$ requires only $n\lceil\log \sigma \rceil (1+o(1))$ bits of space and can answer rank and select queries in $\Theta(\log \sigma)$ time.
A rank query counts the number of occurrences of a symbol up to a given position in the sequence. A select query finds the position in the sequence of a symbol with a given rank. These queries have applications in, e.g., compression, computational geometry, and pattern matching in the form of the backward search---the backbone of many compressed full-text indices.
This repository provides a very fast implementation of wavelet trees in Rust. A companion C++ implementation is available here. More precisely, we implement here a variant called wavelet matrix [2], which gives a more elegant implementation.
The Quad Wavelet Tree (QWT) improves query performance by using a 4-ary tree instead of a binary tree as the basis of the wavelet tree. The 4-ary tree layout of a wavelet tree helps to halve the number of cache misses during queries and thus reduces the query latency.
An experimental evaluation shows that the quad wavelet tree improves the latency of access, rank and select queries by a factor of $\approx$ 2 compared to other implementations of wavelet trees (e.g., the implementation in the widely used C++ Succinct Data Structure Library (SDSL)). For more details, see Benchmarks and the paper [3].
We report here a few experiments to compare our implementation with other state-of-the-art implementations. The experiments are performed using a single thread on a server machine with 8 Intel i9-9900KF cores with base frequencies of 3.60 GHz running Linux 5.19.0. The code is compiled with Rust 1.69.0. Each core has a dedicated L1 cache of size 32 KiB, a dedicated L2 cache of size 256 KiB, a shared L3 cache of size 16 MiB, and 64 GiB of RAM.
A more detailed experimental evaluation can be found in [3].
The dataset is english.2GiB
: the 2 GiB prefix of the English collection from Pizza&Chili corpus (details below). The text has an alphabet with 239 distinct symbols.
| Implementation | access (ns) | rank (ns) | select (ns) | space (MiB) | Language | | :-------------------------------------------- | ------------: | ----------: | ------------: | ----------: | :---------- | | SDSL 2.1.1 | 693 | 786 | 2619 | 3039 | C++ | | Pasta | 832 | 846 | 2403 | 2124 | C++ | | Sucds 0.6.0 | 768 | 818 | 2533 | 2688 | Rust | | QWT 256 | 436 | 441 | 1135 | 2308 | C++/Rust | | QWT 512 | 451 | 460 | 1100 | 2180 | C++/Rust |
We note that the results for the rank query depend on how we generate the symbols to rank in the query set. Here for every rank query, we choose a symbol at random by following the distribution of symbols in the text, i.e., more frequent symbols are selected more frequently. All the data structures have more or less the same performance in ranking rare symbols. The reason is that the portion of the last layers for those rare symbols is likely to fit in the cache.
To run the experiments, we need to compile the binary executables with
bash
cargo build --release
This produces the two executables perf_rs_quat_vector
and perf_wavelet_tree
in \target\release\
.
The former is used to measure the performance of QuadVectors, which are the building block of our implementation of Wavelet Trees. We can safely ignore it.
The latter is used to measure the performance of a Quad Wavelet Tree built on a given input text.
We can now download and uncompress in the current directory the English collection from Pizza&Chili corpus. Then, we take its prefix of length 2 GiB.
bash
wget http://pizzachili.dcc.uchile.cl/texts/nlang/english.gz
gunzip english.gz
head -c 2147483648 english > english.2GiB
The following command builds the wavelet trees (QWT 256 and 512) on this input text and runs 1 million random access, rank, and select queries.
bash
./target/release/perf_wavelet_tree --input-file english.2GiB --access --rank --select
We can use the flag --test-correctness
to perform some extra tests for the correctness of the index. We can also specify the number of queries with n_queries
.
The code measures the latency of the queries by forcing the input of each query to depend on the output of the previous one. This is consistent with the use of the queries in a real setting. For example, the more advanced queries supported by compressed text indexes (e.g., CSA or FM-index) decompose into several dependent queries on the underlying wavelet tree.
Run the following Cargo command in your project directory
cargo add qwt
to add the library.
Once the crate has been added, we can easily build a Quad Wavelet Tree with the following code.
```rust use qwt::QWT256;
let data = vec![1u8, 0, 1, 0, 2, 4, 5, 3];
let qwt = QWT256::from(data);
assert_eq!(qwt.len(), 8); ```
We can print the space usage of the wavelet tree with
```rust use qwt::SpaceUsage;
println!( qwt.spaceusagebytes() ); ```
A wavelet tree implements FromIterator
and, thus, we can use .collect()
.
```rust use qwt::QWT256;
let qwt: QWT256<_> = (0..10u8).intoiter().cycle().take(1000).collect();
assert_eq!(qwt.len(), 1000); ```
The data structure supports three operations:
- get(i)
accesses the i
-th symbols of the indexed sequence;
- rank(c, i)
counts the number of occurrences of symbol c
up to position i
excluded;
- select(c, i)
returns the position of the i
-th occurrence of symbol c
.
Here is an example of the three operations.
```rust use qwt::{QWT256, AccessUnsigned, RankUnsigned, SelectUnsigned};
let data = vec![1u8, 0, 1, 0, 2, 4, 5, 3];
let qwt = QWT256::from(data);
asserteq!(qwt.get(2), Some(1)); asserteq!(qwt.get(3), Some(0)); assert_eq!(qwt.get(8), None);
asserteq!(qwt.rank(1, 2), Some(1)); asserteq!(qwt.rank(1, 0), Some(0)); asserteq!(qwt.rank(3, 8), Some(1)); asserteq!(qwt.rank(1, 9), None);
asserteq!(qwt.select(1, 1), Some(0)); asserteq!(qwt.select(0, 2), Some(3)); asserteq!(qwt.select(4, 0), None); asserteq!(qwt.select(1, 3), None); ```
In the following example, we use QWT to index a sequence over a larger alphabet.
```rust use qwt::{QWT256, AccessUnsigned, RankUnsigned, SelectUnsigned};
let data = vec![1u32, 0, 1, 0, 2, 1000000, 5, 3]; let qwt = QWT256::from(data);
asserteq!(qwt.get(2), Some(1)); asserteq!(qwt.get(5), Some(1000000)); assert_eq!(qwt.get(8), None); ```
For more details, take a look at the documentation.
Serialization and deserialization can be done with bincode
as follows.
```rust use std::fs; use std::path::Path;
use qwt::{QWT256, AccessUnsigned};
let data = vec![1u8, 0, 1, 0, 2, 4, 5, 3]; let qwt = QWT256::from(data);
assert_eq!(qwt.get(2), Some(1));
// serialize let serialized = bincode::serialize(&qwt).unwrap();
// write on file, if needed let outputfilename = "example.qwt256".tostring(); fs::write(Path::new(&output_filename), serialized).unwrap();
// read from file let serialized = fs::read(Path::new(&output_filename)).unwrap();
// deserialize
let qwt = bincode::deserialize::
assert_eq!(qwt.get(2), Some(1));
```
We can index any sequence over any num::traits::Unsigned integers. As the space usage depends on the largest value in the sequence, it could be worth remapping the values to remove "holes".
Please cite the following paper if you use this code.
@misc{QWT,
author = {Matteo Ceregini, Florian Kurpicz, Rossano Venturini},
title = {Faster Wavelet Trees with Quad Vectors},
publisher = {arXiv},
year = {2023},
doi = {10.48550/ARXIV.2302.09239},
url = {http://arxiv.org/abs/2302.09239}
}