#english-words #numbers #scale #convert #short #thousands #power

shortscale

Convert numbers into English words using the short scale

14 stable releases

1.5.1 Sep 3, 2024
1.5.0 Apr 17, 2021
1.4.0 Feb 13, 2021
1.3.2 Jan 10, 2021
0.1.0 Dec 31, 2020

#48 in Value formatting

MIT license

24KB
430 lines

shortscale

CI
Rust docs | crates.io

Rust lib to convert numbers into English words.

This module was written as an exploration of JavaScript and Rust documented here.

The short scale, has different words for each power of 1000.

This library expresses numbers from zero to thousands, millions, billions, trillions, and quadrillions, up to 999_999_999_999_999_999.

Function

pub fn shortscale(num: u64) -> String

Example

use shortscale::shortscale;

assert_eq!(
    shortscale(420_000_999_015),
    "four hundred and twenty billion nine hundred \
    and ninety nine thousand and fifteen"
);

String writer

For efficient writing into a mutable pre-allocated string.
The performance difference is small and varies across systems.

pub fn shortscale_string_writer(s: &mut String, num: u64)
use shortscale::shortscale_string_writer;

let mut my_string = String::from("The number 27 in words is ");
my_string.reserve(1024); // pre-allocate capacity (for performance only)

shortscale_string_writer(&mut my_string, 27);
assert_eq!(my_string, "The number 27 in words is twenty seven");

Extra

As a record of my first foray into rust, older implementations are preserved under shortscale::extra.

For benchmarks run RUSTFLAGS="--cfg extra" cargo bench

GitHub Actions, running on Ubuntu.

test a_shortscale                        ... bench:         262 ns/iter (+/- 42)
test b_shortscale_string_writer_no_alloc ... bench:          99 ns/iter (+/- 17)
test c_str_push                          ... bench:         266 ns/iter (+/- 55)
test d_vec_push                          ... bench:         301 ns/iter (+/- 48)
test e_display_no_alloc                  ... bench:         254 ns/iter (+/- 55)
test f_vec_concat                        ... bench:         635 ns/iter (+/- 131)
test g_string_join                       ... bench:         632 ns/iter (+/- 80)

On MacOS Catalina 2.6 GHz Intel Core i7 memory allocation appears to be a lot slower.

test a_shortscale                        ... bench:         378 ns/iter (+/- 59)
test b_shortscale_string_writer_no_alloc ... bench:          75 ns/iter (+/- 2)
test c_str_push                          ... bench:         396 ns/iter (+/- 42)
test d_vec_push                          ... bench:         435 ns/iter (+/- 12)
test e_display_no_alloc                  ... bench:         217 ns/iter (+/- 30)
test f_vec_concat                        ... bench:       2,087 ns/iter (+/- 79)
test g_string_join                       ... bench:       2,166 ns/iter (+/- 225)

JavaScript

Running npm run bench on the JavaScript version at jldec/shortscale shows that JavaScript is really fast as well - faster on MacOS than my first 2 naive rust implementations.

Ubuntu, Node v14

20000 calls, 1200000 bytes, 1373 ns/call
20000 calls, 1200000 bytes, 1336 ns/call
20000 calls, 1200000 bytes, 1480 ns/call

MacOS

20000 calls, 1200000 bytes, 967 ns/call
20000 calls, 1200000 bytes, 982 ns/call
20000 calls, 1200000 bytes, 965 ns/call

[!NOTE] More recent rust versions will complain about unexpected "cfg" condition name: "extra".
This has been mitigated using the lints.rust unexpected_cfgs in Cargo.toml.

Copyright 2021-2024, Jürgen Leschner - github.com/jldec - MIT license

No runtime deps