<script src="https://cdn.jsdelivr.net/npm/big.js@6.2.2/big.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/fraction.js@5.2.1/dist/fraction.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/number-precision@1.6.0/build/index.iife.min.js"></script>
/*your preparation JavaScript code goes here
To execute async code during the script preparation, wrap it as function globalMeasureThatScriptPrepareFunction, example:*/
async function globalMeasureThatScriptPrepareFunction() {
// This function is optional, feel free to remove it.
// await someThing();
}
var a = 0.1;
var b = 0.2;
((a + b) * b).toString();
var a = 0.1;
var b = 0.2;
((a + b) * b).toFixed(9);
var a = new Big("0.1");
var b = new Big("0.2");
a.plus(b).mul(b).toString();
var a = new Fraction("0.1");
var b = new Fraction("0.2");
a.add(b).mul(b).toString();
var a = 0.1;
var b = 0.2;
NP.times(NP.plus(a, b), b).toString()
--enable-precise-memory-info
flag.
Test case name | Result |
---|---|
Native | |
Native + toFixed | |
Big.js | |
Fraction.js | |
number-precision |
Test name | Executions per second |
---|---|
Native | 99127320.0 Ops/sec |
Native + toFixed | 16131546.0 Ops/sec |
Big.js | 3591213.8 Ops/sec |
Fraction.js | 1802781.9 Ops/sec |
number-precision | 749636.1 Ops/sec |
The benchmark you provided compares the performance of various approaches for performing arithmetic operations involving decimal numbers in JavaScript, specifically testing the addition and multiplication of decimal values 0.1 and 0.2. The different approaches include native JavaScript calculations along with the use of three external libraries: Big.js, Fraction.js, and number-precision.
Native
var a = 0.1;
var b = 0.2;
((a + b) * b).toString();
a
and b
, multiplies the result by b
, and converts the final result to a string.0.1 + 0.2 !== 0.3
.Native + toFixed
var a = 0.1;
var b = 0.2;
((a + b) * b).toFixed(9);
toFixed()
.toFixed()
can help with display issues, it does not fix the underlying floating-point precision problem in calculations. Performance is slower compared to the first approach.Big.js
var a = new Big("0.1");
var b = new Big("0.2");
a.plus(b).mul(b).toString();
Fraction.js
var a = new Fraction("0.1");
var b = new Fraction("0.2");
a.add(b).mul(b).toString();
0.1
becomes 1/10
).number-precision
var a = 0.1;
var b = 0.2;
NP.times(NP.plus(a, b), b).toString();
The benchmark results indicate that the Native approach performed the best with approximately 99 million executions per second. The use of toFixed()
reduced the performance substantially to about 16 million executions per second.
The libraries showed significantly lower performance. Big.js executed around 3.5 million times per second, Fraction.js around 1.8 million, and number-precision at approximately 749,000. This highlights the trade-off between performance and precision when using external libraries.
Precision vs Performance: There's a clear trade-off with libraries like Big.js and Fraction.js, which prioritize precision over performance. While using native types is faster, it introduces potential inaccuracies in calculations that can lead to subtle bugs, especially evident in financial applications.
Alternatives: Besides these libraries, there are other libraries and strategies in JavaScript for dealing with decimal precision. For example, libraries like decimal.js and bignumber.js provide similar functionality with different implementations and performance characteristics.
In conclusion, the choice between native operations and libraries depends heavily on the specific application needs regarding performance requirements and precision handling. For applications requiring high precision and correctness in decimal arithmetic, using libraries is advisable despite the performance hit, while for less critical applications, native operations may suffice.