Improve `to_str_radix` implementation
Uses in-place division with `u32` divisor to provide a significant speed improvement.
This commit is contained in:
parent
d9b72a3366
commit
786541bd24
189
src/bigint.rs
189
src/bigint.rs
|
@ -73,7 +73,7 @@ use std::{i64, u64};
|
||||||
use rand::Rng;
|
use rand::Rng;
|
||||||
use rustc_serialize::hex::ToHex;
|
use rustc_serialize::hex::ToHex;
|
||||||
|
|
||||||
use traits::{ToPrimitive, FromPrimitive, cast};
|
use traits::{ToPrimitive, FromPrimitive};
|
||||||
|
|
||||||
use {Num, Unsigned, CheckedAdd, CheckedSub, CheckedMul, CheckedDiv, Signed, Zero, One};
|
use {Num, Unsigned, CheckedAdd, CheckedSub, CheckedMul, CheckedDiv, Signed, Zero, One};
|
||||||
use self::Sign::{Minus, NoSign, Plus};
|
use self::Sign::{Minus, NoSign, Plus};
|
||||||
|
@ -179,7 +179,7 @@ impl hash::Hash for BigUint {
|
||||||
|
|
||||||
impl fmt::Display for BigUint {
|
impl fmt::Display for BigUint {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
write!(f, "{}", to_str_radix(self, 10))
|
write!(f, "{}", self.to_str_radix(10))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -809,100 +809,68 @@ impl_to_biguint!(u16, FromPrimitive::from_u16);
|
||||||
impl_to_biguint!(u32, FromPrimitive::from_u32);
|
impl_to_biguint!(u32, FromPrimitive::from_u32);
|
||||||
impl_to_biguint!(u64, FromPrimitive::from_u64);
|
impl_to_biguint!(u64, FromPrimitive::from_u64);
|
||||||
|
|
||||||
// Cribbed from core/fmt/num.rs
|
fn to_str_radix_reversed(u: &BigUint, radix: u32) -> Vec<u8> {
|
||||||
#[derive(Copy, Clone)]
|
if radix < 2 || radix > 36 {
|
||||||
pub struct RadixFmt {
|
panic!("invalid radix: {}", radix);
|
||||||
data: BigDigit,
|
}
|
||||||
base: u8
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RadixFmt {
|
if u.is_zero() {
|
||||||
fn digit(&self, x: u8) -> u8 {
|
vec![b'0']
|
||||||
match x {
|
|
||||||
x @ 0 ... 9 => b'0' + x,
|
|
||||||
x if x < self.base => b'a' + (x - 10),
|
|
||||||
x => panic!("number not in the range 0..{}: {}", self.base - 1, x),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for RadixFmt {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
// The radix can be as low as 2, so we need a buffer of at least 64
|
|
||||||
// characters for a base 2 number.
|
|
||||||
let mut x = self.data;
|
|
||||||
let zero = 0;
|
|
||||||
let is_positive = x >= zero;
|
|
||||||
let mut buf = [0u8; 64];
|
|
||||||
let mut curr = buf.len();
|
|
||||||
let base = self.base as BigDigit;
|
|
||||||
if is_positive {
|
|
||||||
// Accumulate each digit of the number from the least significant
|
|
||||||
// to the most significant figure.
|
|
||||||
for byte in buf.iter_mut().rev() {
|
|
||||||
let n = x % base; // Get the current place value.
|
|
||||||
x = x / base; // Deaccumulate the number.
|
|
||||||
*byte = self.digit(cast(n).unwrap()); // Store the digit in the buffer.
|
|
||||||
curr -= 1;
|
|
||||||
if x == zero { break }; // No more digits left to accumulate.
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Do the same as above, but accounting for two's complement.
|
let mut res = Vec::new();
|
||||||
for byte in buf.iter_mut().rev() {
|
let mut digits = u.data.to_vec();
|
||||||
let n = zero - (x % base); // Get the current place value.
|
let mut n_digits = digits.len();
|
||||||
x = x / base; // Deaccumulate the number.
|
|
||||||
*byte = self.digit(cast(n).unwrap()); // Store the digit in the buffer.
|
while n_digits != 0 {
|
||||||
curr -= 1;
|
let rem = div_rem_in_place(&mut digits[..n_digits], radix);
|
||||||
if x == zero { break }; // No more digits left to accumulate.
|
res.push(to_digit(rem as u8));
|
||||||
|
n_digits = count_non_zero(&digits[..n_digits]);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
let buf = unsafe { str::from_utf8_unchecked(&buf[curr..]) };
|
res
|
||||||
f.pad_integral(is_positive, "", buf)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_str_radix(me: &BigUint, radix: u32) -> String {
|
fn div_rem_in_place(digits: &mut [BigDigit], divisor: BigDigit) -> BigDigit {
|
||||||
assert!(1 < radix && radix <= 16, "The radix must be within (1, 16]");
|
let mut rem = 0;
|
||||||
let (base, max_len) = get_radix_base(radix);
|
|
||||||
if base == big_digit::BASE {
|
|
||||||
return fill_concat(&me.data, radix, max_len)
|
|
||||||
}
|
|
||||||
return fill_concat(&convert_base(me, base), radix, max_len);
|
|
||||||
|
|
||||||
fn convert_base(n: &BigUint, base: DoubleBigDigit) -> Vec<BigDigit> {
|
for d in digits.iter_mut().rev() {
|
||||||
let divider = base.to_biguint().unwrap();
|
let (q, r) = full_div_rem(*d, divisor, rem);
|
||||||
let mut result = Vec::new();
|
*d = q;
|
||||||
let mut m = n.clone();
|
rem = r;
|
||||||
while m >= divider {
|
|
||||||
let (d, m0) = m.div_mod_floor(÷r);
|
|
||||||
result.push(m0.to_usize().unwrap() as BigDigit);
|
|
||||||
m = d;
|
|
||||||
}
|
|
||||||
if !m.is_zero() {
|
|
||||||
result.push(m.to_usize().unwrap() as BigDigit);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fill_concat(v: &[BigDigit], radix: u32, l: usize) -> String {
|
rem
|
||||||
if v.is_empty() {
|
|
||||||
return "0".to_string()
|
|
||||||
}
|
|
||||||
let mut s = String::with_capacity(v.len() * l);
|
|
||||||
for n in v.iter().rev() {
|
|
||||||
let ss = format!("{}", RadixFmt { data: *n, base: radix as u8 });
|
|
||||||
s.extend(repeat("0").take(l - ss.len()));
|
|
||||||
s.push_str(&ss);
|
|
||||||
}
|
|
||||||
s.trim_left_matches('0').to_string()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_str_radix_signed(me: &BigInt, radix: u32) -> String {
|
fn count_non_zero(digits: &[u32]) -> usize {
|
||||||
match me.sign {
|
let mut n = digits.len();
|
||||||
Plus => to_str_radix(&me.data, radix),
|
|
||||||
NoSign => "0".to_string(),
|
for &i in digits.iter().rev() {
|
||||||
Minus => format!("-{}", to_str_radix(&me.data, radix)),
|
if i == 0 {
|
||||||
|
n -= 1;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n
|
||||||
|
}
|
||||||
|
|
||||||
|
fn full_div_rem(a: BigDigit, b: BigDigit, borrow: BigDigit) -> (BigDigit, BigDigit) {
|
||||||
|
let lo = a as DoubleBigDigit;
|
||||||
|
let hi = borrow as DoubleBigDigit;
|
||||||
|
|
||||||
|
let lhs = lo | (hi << big_digit::BITS);
|
||||||
|
let rhs = b as DoubleBigDigit;
|
||||||
|
((lhs / rhs) as BigDigit, (lhs % rhs) as BigDigit)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn to_digit(b: u8) -> u8 {
|
||||||
|
match b {
|
||||||
|
0 ... 9 => b'0' + b,
|
||||||
|
10 ... 35 => b'a' - 10 + b,
|
||||||
|
_ => panic!("invalid digit: {}", b)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1013,6 +981,24 @@ impl BigUint {
|
||||||
v
|
v
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the integer formatted as a string in the given radix.
|
||||||
|
/// `radix` must be in the range `[2, 36]`.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use num::bigint::BigUint;
|
||||||
|
///
|
||||||
|
/// let i = BigUint::parse_bytes(b"ff", 16).unwrap();
|
||||||
|
/// assert_eq!(i.to_str_radix(16), "ff");
|
||||||
|
/// ```
|
||||||
|
#[inline]
|
||||||
|
pub fn to_str_radix(&self, radix: u32) -> String {
|
||||||
|
let mut v = to_str_radix_reversed(self, radix);
|
||||||
|
v.reverse();
|
||||||
|
unsafe { String::from_utf8_unchecked(v) }
|
||||||
|
}
|
||||||
|
|
||||||
/// Creates and initializes a `BigUint`.
|
/// Creates and initializes a `BigUint`.
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
|
@ -1168,7 +1154,7 @@ impl Default for BigInt {
|
||||||
|
|
||||||
impl fmt::Display for BigInt {
|
impl fmt::Display for BigInt {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
write!(f, "{}", to_str_radix_signed(self, 10))
|
write!(f, "{}", self.to_str_radix(10))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1779,6 +1765,29 @@ impl BigInt {
|
||||||
(self.sign, self.data.to_bytes_be())
|
(self.sign, self.data.to_bytes_be())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the integer formatted as a string in the given radix.
|
||||||
|
/// `radix` must be in the range `[2, 36]`.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use num::bigint::BigInt;
|
||||||
|
///
|
||||||
|
/// let i = BigInt::parse_bytes(b"ff", 16).unwrap();
|
||||||
|
/// assert_eq!(i.to_str_radix(16), "ff");
|
||||||
|
/// ```
|
||||||
|
#[inline]
|
||||||
|
pub fn to_str_radix(&self, radix: u32) -> String {
|
||||||
|
let mut v = to_str_radix_reversed(&self.data, radix);
|
||||||
|
|
||||||
|
if self.is_negative() {
|
||||||
|
v.push(b'-');
|
||||||
|
}
|
||||||
|
|
||||||
|
v.reverse();
|
||||||
|
unsafe { String::from_utf8_unchecked(v) }
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns the sign of the `BigInt` as a `Sign`.
|
/// Returns the sign of the `BigInt` as a `Sign`.
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
|
@ -1874,7 +1883,7 @@ impl From<ParseIntError> for ParseBigIntError {
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod biguint_tests {
|
mod biguint_tests {
|
||||||
use Integer;
|
use Integer;
|
||||||
use super::{BigDigit, BigUint, ToBigUint, to_str_radix, big_digit};
|
use super::{BigDigit, BigUint, ToBigUint, big_digit};
|
||||||
use super::{BigInt, RandBigInt, ToBigInt};
|
use super::{BigInt, RandBigInt, ToBigInt};
|
||||||
use super::Sign::Plus;
|
use super::Sign::Plus;
|
||||||
|
|
||||||
|
@ -2059,7 +2068,7 @@ mod biguint_tests {
|
||||||
fn test_shl() {
|
fn test_shl() {
|
||||||
fn check(s: &str, shift: usize, ans: &str) {
|
fn check(s: &str, shift: usize, ans: &str) {
|
||||||
let opt_biguint = BigUint::from_str_radix(s, 16).ok();
|
let opt_biguint = BigUint::from_str_radix(s, 16).ok();
|
||||||
let bu = to_str_radix(&(opt_biguint.unwrap() << shift), 16);
|
let bu = (opt_biguint.unwrap() << shift).to_str_radix(16);
|
||||||
assert_eq!(bu, ans);
|
assert_eq!(bu, ans);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2180,7 +2189,7 @@ mod biguint_tests {
|
||||||
fn test_shr() {
|
fn test_shr() {
|
||||||
fn check(s: &str, shift: usize, ans: &str) {
|
fn check(s: &str, shift: usize, ans: &str) {
|
||||||
let opt_biguint = BigUint::from_str_radix(s, 16).ok();
|
let opt_biguint = BigUint::from_str_radix(s, 16).ok();
|
||||||
let bu = to_str_radix(&(opt_biguint.unwrap() >> shift), 16);
|
let bu = (opt_biguint.unwrap() >> shift).to_str_radix(16);
|
||||||
assert_eq!(bu, ans);
|
assert_eq!(bu, ans);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2684,7 +2693,7 @@ mod biguint_tests {
|
||||||
let &(ref n, ref rs) = num_pair;
|
let &(ref n, ref rs) = num_pair;
|
||||||
for str_pair in rs.iter() {
|
for str_pair in rs.iter() {
|
||||||
let &(ref radix, ref str) = str_pair;
|
let &(ref radix, ref str) = str_pair;
|
||||||
assert_eq!(to_str_radix(n, *radix), *str);
|
assert_eq!(n.to_str_radix(*radix), *str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue