Quantizer

Simple vector quantization utilities and functions.
cargo add vector_quantizer
Example usage:
use anyhow::Result;
use ndarray::Array2;
use ndarray_rand::RandomExt;
use vector_quantizer::pq::PQ;
use rand_distr::StandardNormal;
fn main() -> Result<()> {
let num_vectors = 1000;
let dimension = 128;
let original_vectors = Array2::random((num_vectors, dimension), StandardNormal);
let m = 8; let ks = 256; let mut pq = PQ::try_new(m, ks, Some(true))?;
println!("Training PQ model...");
pq.fit(&original_vectors, 20)?;
println!("Quantizing vectors...");
let quantized_vectors = pq.compress(&original_vectors)?;
let compression_ratio = calc_compression_ratio(m, ks, dimension);
let mse = calc_mse(&original_vectors, &quantized_vectors);
println!("\nQuantization Results:");
println!("Original vector size: {} bytes", dimension * 4); println!("Quantized vector size: {} bytes", m); println!("Compression ratio: {:.2}x", compression_ratio);
println!("Mean Squared Error: {:.6}", mse);
let compact_codes = pq.encode(&original_vectors)?;
println!("\nCompact codes shape: {:?}", compact_codes.dim());
let reconstructed = pq.decode(&compact_codes)?;
assert_eq!(reconstructed.dim(), original_vectors.dim());
Ok(())
}
fn calc_compression_ratio(m: usize, ks: u32, dimension: usize) -> f64 {
let original_size = dimension * 4; let quantized_size = m; original_size as f64 / quantized_size as f64
}
fn calc_mse(original: &Array2<f32>, quantized: &Array2<f32>) -> f32 {
(&(original - quantized))
.mapv(|x| x.powi(2))
.mean()
.unwrap()
}
See a more detailed example here: /src/bin/example.rs
Acknowledgements
The code in this repository is mostly adapted from https://github.com/xinyandai/product-quantization, a great Python lib for vector quantization.
The original code and the one written in this repository is derived from "Norm-Explicit Quantization: Improving Vector Quantization for Maximum Inner Product Search" by Dai, Xinyan and Yan, Xiao and Ng, Kelvin KW and Liu, Jie and Cheng, James: https://arxiv.org/abs/1911.04654