Skip to content
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions DIRECTORY.md
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,7 @@
* [CheckKishnamurthyNumber](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/CheckKishnamurthyNumber.js)
* [Coordinate](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/Coordinate.js)
* [CoPrimeCheck](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/CoPrimeCheck.js)
* [DecimalExpansion](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/DecimalExpansion.js)
* [decimalIsolate](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/decimalIsolate.js)
* [DegreeToRadian](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/DegreeToRadian.js)
* [DigitSum](https://github.com/TheAlgorithms/Javascript/blob/master/Maths/DigitSum.js)
Expand Down
68 changes: 68 additions & 0 deletions Maths/DecimalExpansion.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
/**
* @author Eric Lavault <https://github.com/lvlte>
* @license MIT
*
* Represents the decimal (or binary, octal, any base from 2 to 10) expansion
* of a/b using euclidean division.
*
* Because this function is recursive, it may throw an error when reaching the
* maximum call stack size.
*
* Returns an array containing : [
* 0: integer part of the division
* 1: array of decimals (if any, or an empty array)
* 2: indexOf 1st cycle digit in decimals array if a/b is periodic, or undef.
* ]
*
* @see https://mathworld.wolfram.com/DecimalExpansion.html
*
* @param {number} a
* @param {number} b
* @param {number} [base=10]
* @returns {array}
*/
export function decExp (a, b, base = 10, exp = [], d = {}, dlen = 0) {
if (base < 2 || base > 10) {
throw new RangeError('Unsupported base. Must be in range [2, 10]')
}

if (a === 0) {
return [0, [], undefined]
}

if (a === b && dlen === 0) {
return [1, [], undefined]
}

// d contains the dividends used so far and the corresponding index of its
// euclidean division by b in the expansion array.
d[a] = dlen++

if (a < b) {
exp.push(0)
return decExp(a * base, b, base, exp, d, dlen)
}

// Euclid's division lemma : a = bq + r
const r = a % b
const q = (a - r) / b

// Decimal expansion (1st element is the integer part)
exp.push(+q.toString(base))

if (r === 0) {
// got a regular number (division terminates)
return [exp[0], exp.slice(1), undefined]
}

// For the next iteration
a = r * base

// Check if `a` has already been used as a dividend, in which case it means
// the expansion is periodic.
if (a in d) {
return [exp[0], exp.slice(1), d[a] - 1]
}

return decExp(a, b, base, exp, d, dlen)
}
192 changes: 192 additions & 0 deletions Maths/test/DecimalExpansion.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,192 @@
import { decExp } from '../DecimalExpansion'

/**
* Decimal
*/

describe('Finite Decimal Expansion', () => {
it('1/2 = 0.5', () => {
const [integer, decimals, cycleIndex] = decExp(1, 2)
expect(integer).toBe(0)
expect(decimals).toEqual([5])
expect(cycleIndex).toBeUndefined()
})

it('1/5 = 0.2', () => {
const [integer, decimals, cycleIndex] = decExp(1, 5)
expect(integer).toBe(0)
expect(decimals).toEqual([2])
expect(cycleIndex).toBeUndefined()
})

it('1/8 = 0.125', () => {
const [integer, decimals, cycleIndex] = decExp(1, 8)
expect(integer).toBe(0)
expect(decimals).toEqual([1, 2, 5])
expect(cycleIndex).toBeUndefined()
})

it('255/40 = 6.375', () => {
const [integer, decimals, cycleIndex] = decExp(255, 40)
expect(integer).toBe(6)
expect(decimals).toEqual([3, 7, 5])
expect(cycleIndex).toBeUndefined()
})
})

describe('Repeating Decimal Expansion', () => {
it('1/3 = 0.(3)', () => {
expect(decExp(1, 3)).toStrictEqual([0, [3], 0])
})

it('1/6 = 0.1(6)', () => {
expect(decExp(1, 6)).toStrictEqual([0, [1, 6], 1])
})

it('1/7 = 0.(142857)', () => {
expect(decExp(1, 7)).toStrictEqual([0, [1, 4, 2, 8, 5, 7], 0])
})
})

/**
* Binary
*/

describe('Finite Binary Expansion', () => {
it('1/2 = 0.1₂', () => {
const [integer, decimals, cycleIndex] = decExp(1, 2, 2)
expect(integer).toBe(0)
expect(decimals).toEqual([1])
expect(cycleIndex).toBeUndefined()
})

it('1/8 = 0.001₂', () => {
const [integer, decimals, cycleIndex] = decExp(1, 8, 2)
expect(integer).toBe(0)
expect(decimals).toEqual([0, 0, 1])
expect(cycleIndex).toBeUndefined()
})

it('255/40 = 110.011₂', () => {
const [integer, decimals, cycleIndex] = decExp(255, 40, 2)
expect(integer).toBe(110)
expect(decimals).toEqual([0, 1, 1])
expect(cycleIndex).toBeUndefined()
})
})

describe('Repeating Binary Expansion', () => {
it('1/3 = 0.(01)₂', () => {
expect(decExp(1, 3, 2)).toStrictEqual([0, [0, 1], 0])
})

it('1/5 = 0.(0011)₂', () => {
expect(decExp(1, 5, 2)).toStrictEqual([0, [0, 0, 1, 1], 0])
})

it('1/6 = 0.0(01)₂', () => {
expect(decExp(1, 6, 2)).toStrictEqual([0, [0, 0, 1], 1])
})

it('1/7 = 0.(001)₂', () => {
expect(decExp(1, 7, 2)).toStrictEqual([0, [0, 0, 1], 0])
})
})

/**
* Octal
*/

describe('Finite Octal Expansion', () => {
it('1/2 = 0.4₈', () => {
const [integer, decimals, cycleIndex] = decExp(1, 2, 8)
expect(integer).toBe(0)
expect(decimals).toEqual([4])
expect(cycleIndex).toBeUndefined()
})

it('1/8 = 0.1₈', () => {
const [integer, decimals, cycleIndex] = decExp(1, 8, 8)
expect(integer).toBe(0)
expect(decimals).toEqual([1])
expect(cycleIndex).toBeUndefined()
})

it('255/40 = 6.3₈', () => {
const [integer, decimals, cycleIndex] = decExp(255, 40, 8)
expect(integer).toBe(6)
expect(decimals).toEqual([3])
expect(cycleIndex).toBeUndefined()
})
})

describe('Repeating Octal Expansion', () => {
it('1/3 = 0.(25)₈', () => {
expect(decExp(1, 3, 8)).toStrictEqual([0, [2, 5], 0])
})

it('1/5 = 0.(1463)₈', () => {
expect(decExp(1, 5, 8)).toStrictEqual([0, [1, 4, 6, 3], 0])
})

it('1/6 = 0.1(25)₈', () => {
expect(decExp(1, 6, 8)).toStrictEqual([0, [1, 2, 5], 1])
})

it('1/7 = 0.(1)₈', () => {
expect(decExp(1, 7, 8)).toStrictEqual([0, [1], 0])
})
})

/**
* Integers
*/

describe('Integers', () => {
it('1/1 = 1', () => {
const [integer, decimals, cycleIndex] = decExp(1, 1)
expect(integer).toBe(1)
expect(decimals).toStrictEqual([])
expect(cycleIndex).toBeUndefined()
})

it('5/5 = 1', () => {
const [integer, decimals, cycleIndex] = decExp(5, 5)
expect(integer).toBe(1)
expect(decimals).toStrictEqual([])
expect(cycleIndex).toBeUndefined()
})

it('2/1 = 2', () => {
const [integer, decimals, cycleIndex] = decExp(2, 1)
expect(integer).toBe(2)
expect(decimals).toStrictEqual([])
expect(cycleIndex).toBeUndefined()
})

it('9/3 = 3', () => {
const [integer, decimals, cycleIndex] = decExp(9, 3)
expect(integer).toBe(3)
expect(decimals).toStrictEqual([])
expect(cycleIndex).toBeUndefined()
})
})

/**
* Unsupported base
*/

describe('Throws on unsupported base', () => {
it('negative base', () => {
expect(() => decExp(1, 1, -2)).toThrow(RangeError)
})
it('base 0', () => {
expect(() => decExp(1, 1, 0)).toThrow(RangeError)
})
it('base 1', () => {
expect(() => decExp(1, 1, 1)).toThrow(RangeError)
})
it('base 11', () => {
expect(() => decExp(1, 1, 11)).toThrow(RangeError)
})
})