-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathfibonacci.ts
71 lines (60 loc) · 2.14 KB
/
fibonacci.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
import { OpenFeature } from '@openfeature/nestjs-sdk';
const oFeatClient = OpenFeature.getClient('fibonacci');
export async function fibonacci(num: number): Promise<number> {
const value = await oFeatClient.getStringValue('fib-algo', 'recursive');
switch (value) {
case 'recursive':
return getNthFibRecursive(num);
case 'memo':
return getNthFibRecursiveMemo(num);
case 'loop':
return getNthFibLoop(num);
case 'binet':
return getNthFibBinet(num);
default:
return getNthFibRecursive(num);
}
}
export async function getNthFibRecursive(num: number): Promise<number> {
// on very fast systems, recursive is faster than we want for the demo... so we add an artificial linear delay, based on n.
const minTimePromise = new Promise((resolve) => setTimeout(resolve, Math.floor((num / 10) * 1000)));
const [result] = await Promise.all([getNthFibRecursiveSync(num), minTimePromise]);
return result;
}
function getNthFibRecursiveSync(num: number): number {
if (num === 2) {
return 1;
} else if (num === 1) {
return 0;
} else {
return getNthFibRecursiveSync(num - 1) + getNthFibRecursiveSync(num - 2);
}
}
type Cache = { [num: number]: number };
export function getNthFibRecursiveMemo(num: number, memo: Cache = { 1: 0, 2: 1 }): number {
return getNthFibRecursiveMemoSync(num, memo);
}
function getNthFibRecursiveMemoSync(num: number, memo: Cache = { 1: 0, 2: 1 }): number {
if (num in memo) {
return memo[num];
} else {
memo[num] = getNthFibRecursiveMemoSync(num - 1, memo) + getNthFibRecursiveMemoSync(num - 2, memo);
return memo[num];
}
}
export function getNthFibLoop(num: number): number {
const previousTwoNum: [number, number] = [0, 1];
let counter = 3;
while (counter <= num) {
const nextNum = previousTwoNum[0] + previousTwoNum[1];
previousTwoNum[0] = previousTwoNum[1];
previousTwoNum[1] = nextNum;
counter++;
}
return num > 1 ? previousTwoNum[1] : previousTwoNum[0];
}
export function getNthFibBinet(num: number): number {
return Math.round(
(Math.pow((1 + Math.sqrt(5)) / 2, num - 1) - Math.pow((1 - Math.sqrt(5)) / 2, num - 1)) / Math.sqrt(5)
);
}