1
0
mirror of https://github.com/MikeMcl/decimal.js.git synced 2024-09-28 22:40:48 +00:00
MikeMcl_decimal.js/README.md

247 lines
8.1 KiB
Markdown
Raw Normal View History

2014-04-02 15:28:08 +00:00
![decimal.js](https://raw.githubusercontent.com/MikeMcl/decimal.js/gh-pages/decimaljs.png)
An arbitrary-precision Decimal type for JavaScript.
2016-02-04 23:52:52 +00:00
2020-09-28 21:28:15 +00:00
[![npm version](https://img.shields.io/npm/v/decimal.js.svg)](https://www.npmjs.com/package/decimal.js)
[![npm downloads](https://img.shields.io/npm/dw/decimal.js)](https://www.npmjs.com/package/decimal.js)
2014-11-11 10:44:42 +00:00
[![Build Status](https://travis-ci.org/MikeMcl/decimal.js.svg)](https://travis-ci.org/MikeMcl/decimal.js)
[![CDNJS](https://img.shields.io/cdnjs/v/decimal.js.svg)](https://cdnjs.com/libraries/decimal.js)
2017-05-04 12:05:38 +00:00
2016-02-29 19:24:47 +00:00
<br>
2014-04-02 15:28:08 +00:00
## Features
2016-02-29 19:24:47 +00:00
- Integers and floats
2016-01-25 00:11:32 +00:00
- Simple but full-featured API
- Replicates many of the methods of JavaScript's `Number.prototype` and `Math` objects
- Also handles hexadecimal, binary and octal values
2014-04-02 15:28:08 +00:00
- Faster, smaller, and perhaps easier to use than JavaScript versions of Java's BigDecimal
- No dependencies
- Wide platform compatibility: uses JavaScript 1.5 (ECMAScript 3) features only
2021-02-16 12:37:43 +00:00
- Comprehensive [documentation](https://mikemcl.github.io/decimal.js/) and test set
2021-06-25 11:46:39 +00:00
- Used under the hood by [math.js](https://github.com/josdejong/mathjs)
2017-12-03 18:09:41 +00:00
- Includes a TypeScript declaration file: *decimal.d.ts*
2014-04-02 15:28:08 +00:00
![API](https://raw.githubusercontent.com/MikeMcl/decimal.js/gh-pages/API.png)
The library is similar to [bignumber.js](https://github.com/MikeMcl/bignumber.js/), but here
2016-11-09 17:08:38 +00:00
precision is specified in terms of significant digits rather than decimal places, and all
2014-04-02 15:28:08 +00:00
calculations are rounded to the precision (similar to Python's decimal module) rather than just
those involving division.
2016-02-29 19:24:47 +00:00
This library also adds the trigonometric functions, among others, and supports non-integer powers,
which makes it a significantly larger library than *bignumber.js* and the even smaller
[big.js](https://github.com/MikeMcl/big.js/).
2014-04-02 15:28:08 +00:00
2021-06-22 18:20:14 +00:00
For a lighter version of this library without the trigonometric functions see
[decimal.js-light](https://github.com/MikeMcl/decimal.js-light/).
2016-11-09 17:08:38 +00:00
2014-04-02 15:28:08 +00:00
## Load
2020-09-28 21:28:15 +00:00
The library is the single JavaScript file *decimal.js* or ES module *decimal.mjs*.
2014-04-02 15:28:08 +00:00
2017-12-10 18:32:38 +00:00
Browser:
2016-02-29 19:24:47 +00:00
```html
<script src='path/to/decimal.js'></script>
2021-06-25 11:46:39 +00:00
2020-09-28 21:28:15 +00:00
<script type="module">
2021-06-25 11:46:39 +00:00
import Decimal from './path/to/decimal.mjs';
...
2020-09-28 21:28:15 +00:00
</script>
```
2021-02-16 12:37:43 +00:00
[Node.js](https://nodejs.org):
2017-12-10 18:32:38 +00:00
```bash
2021-06-22 18:20:14 +00:00
npm install decimal.js
2017-12-10 18:32:38 +00:00
```
2016-02-29 19:24:47 +00:00
```js
2021-06-25 11:46:39 +00:00
const Decimal = require('decimal.js');
2021-06-22 18:20:14 +00:00
import Decimal from 'decimal.js';
2021-06-25 11:46:39 +00:00
2021-06-22 18:20:14 +00:00
import {Decimal} from 'decimal.js';
```
2016-02-29 19:24:47 +00:00
2014-04-02 15:28:08 +00:00
## Use
2021-06-25 11:46:39 +00:00
*In all examples below, semicolons and `toString` calls are not shown.
2014-04-02 15:28:08 +00:00
If a commented-out value is in quotes it means `toString` has been called on the preceding expression.*
2021-06-25 11:46:39 +00:00
The library exports a single constructor function, `Decimal`, which expects a single argument that is a number, string or Decimal instance.
2016-02-29 19:24:47 +00:00
```js
x = new Decimal(123.4567)
y = new Decimal('123456.7e-3')
z = new Decimal(x)
x.equals(y) && y.equals(z) && x.equals(z) // true
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
If using values with more than a few digits, it is recommended to pass strings rather than numbers to avoid a potential loss of precision.
```js
// Precision loss from using numeric literals with more than 15 significant digits.
new Decimal(1.0000000000000001) // '1'
new Decimal(88259496234518.57) // '88259496234518.56'
new Decimal(99999999999999999999) // '100000000000000000000'
// Precision loss from using numeric literals outside the range of Number values.
new Decimal(2e+308) // 'Infinity'
new Decimal(1e-324) // '0'
// Precision loss from the unexpected result of arithmetic with Number values.
new Decimal(0.7 + 0.1) // '0.7999999999999999'
```
As with JavaScript numbers, strings can contain underscores as separators to improve readability.
```js
x = new Decimal('2_147_483_647')
```
String values in binary, hexadecimal or octal notation are also accepted if the appropriate prefix is included.
2016-02-29 19:24:47 +00:00
```js
x = new Decimal('0xff.f') // '255.9375'
y = new Decimal('0b10101100') // '172'
z = x.plus(y) // '427.9375'
2016-02-29 19:24:47 +00:00
z.toBinary() // '0b110101011.1111'
z.toBinary(13) // '0b1.101010111111p+8'
2017-12-03 18:09:41 +00:00
2021-06-25 11:46:39 +00:00
// Using binary exponential notation to create a Decimal with the value of `Number.MAX_VALUE`.
2017-12-03 18:09:41 +00:00
x = new Decimal('0b1.1111111111111111111111111111111111111111111111111111p+1023')
2021-06-25 11:46:39 +00:00
// '1.7976931348623157081e+308'
2017-12-03 18:09:41 +00:00
```
2021-06-25 11:46:39 +00:00
Decimal instances are immutable in the sense that they are not changed by their methods.
2016-02-29 19:24:47 +00:00
```js
0.3 - 0.1 // 0.19999999999999998
x = new Decimal(0.3)
x.minus(0.1) // '0.2'
x // '0.3'
```
2016-02-29 19:24:47 +00:00
2014-04-02 15:28:08 +00:00
The methods that return a Decimal can be chained.
2016-02-29 19:24:47 +00:00
```js
x.dividedBy(y).plus(z).times(9).floor()
2021-06-25 11:46:39 +00:00
x.times('1.23456780123456789e+9').plus(9876.5432321).dividedBy('4444562598.111772').ceil()
```
2016-02-29 19:24:47 +00:00
2014-04-02 15:28:08 +00:00
Many method names have a shorter alias.
2016-02-29 19:24:47 +00:00
```js
2021-06-25 11:46:39 +00:00
x.squareRoot().dividedBy(y).toPower(3).equals(x.sqrt().div(y).pow(3)) // true
x.comparedTo(y.modulo(z).negated() === x.cmp(y.mod(z).neg()) // true
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
Most of the methods of JavaScript's `Number.prototype` and `Math` objects are replicated.
2016-02-29 19:24:47 +00:00
```js
x = new Decimal(255.5)
2021-06-25 11:46:39 +00:00
x.toExponential(5) // '2.55500e+2'
x.toFixed(5) // '255.50000'
x.toPrecision(5) // '255.50'
2016-02-29 19:24:47 +00:00
Decimal.sqrt('6.98372465832e+9823') // '8.3568682281821340204e+4911'
Decimal.pow(2, 0.0979843) // '1.0702770511687781839'
2021-06-25 11:46:39 +00:00
// Using `toFixed()` to avoid exponential notation:
x = new Decimal('0.0000001')
x.toString() // '1e-7'
2021-06-25 11:55:36 +00:00
x.toFixed() // '0.0000001'
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
And there are `isNaN` and `isFinite` methods, as `NaN` and `Infinity` are valid `Decimal` values.
2016-02-29 19:24:47 +00:00
```js
x = new Decimal(NaN) // 'NaN'
y = new Decimal(Infinity) // 'Infinity'
x.isNaN() && !y.isNaN() && !x.isFinite() && !y.isFinite() // true
2016-02-29 19:24:47 +00:00
```
2021-06-25 11:46:39 +00:00
There is also a `toFraction` method with an optional *maximum denominator* argument.
2016-02-29 19:24:47 +00:00
```js
z = new Decimal(355)
pi = z.dividedBy(113) // '3.1415929204'
pi.toFraction() // [ '7853982301', '2500000000' ]
pi.toFraction(1000) // [ '355', '113' ]
```
2016-02-29 19:24:47 +00:00
2021-06-22 18:20:14 +00:00
All calculations are rounded according to the number of significant digits and rounding mode specified
by the `precision` and `rounding` properties of the Decimal constructor.
2016-02-29 19:24:47 +00:00
2021-06-22 18:20:14 +00:00
For advanced usage, multiple Decimal constructors can be created, each with their own independent
configuration which applies to all Decimal numbers created from it.
2016-02-29 19:24:47 +00:00
```js
// Set the precision and rounding of the default Decimal constructor
2016-11-09 17:08:38 +00:00
Decimal.set({ precision: 5, rounding: 4 })
2014-04-02 15:28:08 +00:00
// Create another Decimal constructor, optionally passing in a configuration object
2021-06-25 11:46:39 +00:00
Dec = Decimal.clone({ precision: 9, rounding: 1 })
2014-04-02 15:28:08 +00:00
x = new Decimal(5)
2021-06-25 11:46:39 +00:00
y = new Dec(5)
2014-04-02 15:28:08 +00:00
x.div(3) // '1.6667'
2017-12-10 18:32:38 +00:00
y.div(3) // '1.66666666'
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
The value of a Decimal is stored in a floating point format in terms of its digits, exponent and sign, but these properties should be considered read-only.
2016-02-29 19:24:47 +00:00
```js
x = new Decimal(-12345.67);
2016-02-29 19:24:47 +00:00
x.d // [ 12345, 6700000 ] digits (base 10000000)
x.e // 4 exponent (base 10)
x.s // -1 sign
```
2016-02-29 19:24:47 +00:00
2014-04-02 15:28:08 +00:00
For further information see the [API](http://mikemcl.github.io/decimal.js/) reference in the *doc* directory.
## Test
2021-06-25 11:46:39 +00:00
To run the tests using Node.js from the root directory:
2016-02-29 19:24:47 +00:00
```bash
2021-06-22 18:20:14 +00:00
npm test
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
Each separate test module can also be executed individually, for example:
2016-02-29 19:24:47 +00:00
```bash
2021-06-25 11:46:39 +00:00
node test/modules/toFraction
2016-02-29 19:24:47 +00:00
```
2021-06-25 11:46:39 +00:00
To run the tests in a browser, open *test/test.html*.
2014-04-02 15:28:08 +00:00
2021-06-22 18:20:14 +00:00
## Minify
2014-04-02 15:28:08 +00:00
2021-06-25 11:46:39 +00:00
Two minification examples:
Using [uglify-js](https://github.com/mishoo/UglifyJS) to minify the *decimal.js* file:
2016-02-29 19:24:47 +00:00
```bash
npm install uglify-js -g
2021-06-25 11:46:39 +00:00
uglifyjs decimal.js --source-map url=decimal.min.js.map -c -m -o decimal.min.js
```
2016-02-29 19:24:47 +00:00
2021-06-25 11:46:39 +00:00
Using [terser](https://github.com/terser/terser) to minify the ES module version, *decimal.mjs*:
2016-02-29 19:24:47 +00:00
```bash
2021-06-22 18:20:14 +00:00
npm install terser -g
terser decimal.mjs --source-map url=decimal.min.mjs.map -c -m --toplevel -o decimal.min.mjs
```
2016-02-29 19:24:47 +00:00
2021-06-22 18:20:14 +00:00
```js
import Decimal from './decimal.min.mjs';
```
2014-04-02 15:28:08 +00:00
## Licence
2021-06-25 11:46:39 +00:00
[The MIT Licence (Expat).](LICENCE.md)