Skip to content

Commit bf7794e

Browse files
authored
Optimize NumDigits method (#356)
1 parent 547861c commit bf7794e

File tree

2 files changed

+57
-10
lines changed

2 files changed

+57
-10
lines changed

decimal.go

+25-6
Original file line numberDiff line numberDiff line change
@@ -1224,14 +1224,33 @@ func (d Decimal) Ln(precision int32) (Decimal, error) {
12241224
}
12251225

12261226
// NumDigits returns the number of digits of the decimal coefficient (d.Value)
1227-
// Note: Current implementation is extremely slow for large decimals and/or decimals with large fractional part
12281227
func (d Decimal) NumDigits() int {
1229-
d.ensureInitialized()
1230-
// Note(mwoss): It can be optimized, unnecessary cast of big.Int to string
1231-
if d.IsNegative() {
1232-
return len(d.value.String()) - 1
1228+
if d.value == nil {
1229+
return 1
1230+
}
1231+
1232+
if d.value.IsInt64() {
1233+
i64 := d.value.Int64()
1234+
// restrict fast path to integers with exact conversion to float64
1235+
if i64 <= (1<<53) && i64 >= -(1<<53) {
1236+
if i64 == 0 {
1237+
return 1
1238+
}
1239+
return int(math.Log10(math.Abs(float64(i64)))) + 1
1240+
}
1241+
}
1242+
1243+
estimatedNumDigits := int(float64(d.value.BitLen()) / math.Log2(10))
1244+
1245+
// estimatedNumDigits (lg10) may be off by 1, need to verify
1246+
digitsBigInt := big.NewInt(int64(estimatedNumDigits))
1247+
errorCorrectionUnit := digitsBigInt.Exp(tenInt, digitsBigInt, nil)
1248+
1249+
if d.value.CmpAbs(errorCorrectionUnit) >= 0 {
1250+
return estimatedNumDigits + 1
12331251
}
1234-
return len(d.value.String())
1252+
1253+
return estimatedNumDigits
12351254
}
12361255

12371256
// IsInteger returns true when decimal can be represented as an integer value, otherwise, it returns false.

decimal_bench_test.go

+32-4
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,34 @@ func BenchmarkDecimal_RoundCash_Five(b *testing.B) {
121121
}
122122
}
123123

124+
func numDigits(b *testing.B, want int, val Decimal) {
125+
b.Helper()
126+
for i := 0; i < b.N; i++ {
127+
if have := val.NumDigits(); have != want {
128+
b.Fatalf("\nHave: %d\nWant: %d", have, want)
129+
}
130+
}
131+
}
132+
133+
func BenchmarkDecimal_NumDigits10(b *testing.B) {
134+
numDigits(b, 10, New(3478512345, -3))
135+
}
136+
137+
func BenchmarkDecimal_NumDigits100(b *testing.B) {
138+
s := make([]byte, 102)
139+
for i := range s {
140+
s[i] = byte('0' + i%10)
141+
}
142+
s[0] = '-'
143+
s[100] = '.'
144+
d, err := NewFromString(string(s))
145+
if err != nil {
146+
b.Log(d)
147+
b.Error(err)
148+
}
149+
numDigits(b, 100, d)
150+
}
151+
124152
func Benchmark_Cmp(b *testing.B) {
125153
decimals := DecimalSlice([]Decimal{})
126154
for i := 0; i < 1000000; i++ {
@@ -132,7 +160,7 @@ func Benchmark_Cmp(b *testing.B) {
132160
}
133161
}
134162

135-
func Benchmark_decimal_Decimal_Add_different_precision(b *testing.B) {
163+
func BenchmarkDecimal_Add_different_precision(b *testing.B) {
136164
d1 := NewFromFloat(1000.123)
137165
d2 := NewFromFloat(500).Mul(NewFromFloat(0.12))
138166

@@ -143,7 +171,7 @@ func Benchmark_decimal_Decimal_Add_different_precision(b *testing.B) {
143171
}
144172
}
145173

146-
func Benchmark_decimal_Decimal_Sub_different_precision(b *testing.B) {
174+
func BenchmarkDecimal_Sub_different_precision(b *testing.B) {
147175
d1 := NewFromFloat(1000.123)
148176
d2 := NewFromFloat(500).Mul(NewFromFloat(0.12))
149177

@@ -154,7 +182,7 @@ func Benchmark_decimal_Decimal_Sub_different_precision(b *testing.B) {
154182
}
155183
}
156184

157-
func Benchmark_decimal_Decimal_Add_same_precision(b *testing.B) {
185+
func BenchmarkDecimal_Add_same_precision(b *testing.B) {
158186
d1 := NewFromFloat(1000.123)
159187
d2 := NewFromFloat(500.123)
160188

@@ -165,7 +193,7 @@ func Benchmark_decimal_Decimal_Add_same_precision(b *testing.B) {
165193
}
166194
}
167195

168-
func Benchmark_decimal_Decimal_Sub_same_precision(b *testing.B) {
196+
func BenchmarkDecimal_Sub_same_precision(b *testing.B) {
169197
d1 := NewFromFloat(1000.123)
170198
d2 := NewFromFloat(500.123)
171199

0 commit comments

Comments
 (0)