Topics:
Books:
- Learning JavaScript DSA Second Edition
- Eloquent JavaScript
- DSA With JavaScript
- DSA
- A Common Sense Guide To DSA
We say that an algorithm is O(fn(n)) if, for sufficiently large values of n, the number of simple operations the computer has to perform is bounded above by a constant times fn(n).
- fn(n) could be linear if it's a constant multiple of n (e.g., fn(n) = n).
- fn(n) could be quadratic if it's a constant multiple of n squared (e.g., fn(n) = n^2).
- fn(n) could be constant if the algorithm's runtime is not dependent on the size of n (e.g., fn(n) = 1).
- fn(n) could be something entirely different.
- Arithmetic operations are constant.
- Variable assignment is constant.
- Accessing elements in an [array] by index or {object} by key is constant.
- in a loop, the complexity is the length of the loop times the complexity of whatever happens inside of the loop.
Summary: How many basic steps or operations an algorithm takes to complete its task
Time complexity is a measure of the amount of time an algorithm takes to complete, expressed as a function of the size of the input. It's often described using big O notation, where O(fn(n)) represents the upper bound of the growth rate of the running time.
- O(1): Constant time complexity. The algorithm's runtime is constant, regardless of the input size.
- O(log n): Logarithmic time complexity. Common in algorithms that divide the problem into smaller sub-problems.
- O(n): Linear time complexity. The runtime grows linearly with the size of the input.
- O(n^2): Quadratic time complexity. Common in algorithms with nested loops.
Summary: The amount of resources, specifically memory, that an algorithm uses to solve a problem
Space complexity is a measure of the amount of memory an algorithm uses, expressed as a function of the size of the input. Similar to time complexity, it's often described using big O notation to represent the upper bound of memory usage.
- O(1): Constant space complexity. The amount of memory used is constant, regardless of the input size.
- O(n): Linear space complexity. The amount of memory used grows linearly with the size of the input.
- O(n^2): Quadratic space complexity. The amount of memory used grows quadratically with the size of the input.
It's crucial to consider both time complexity and space complexity when analyzing algorithms to understand their overall efficiency.
-
Primitive Types:
- Most primitive types (Boolean, Number, undefined, null) are Constant space O(1). They have a fixed size in memory.
-
Strings:
- Strings require Linear space O(n), where n is the string length. Each character contributes to the overall space complexity.
-
Reference Types (Arrays and Objects):
- Reference types generally have Linear space O(n), where n is the number of elements in the array or the number of keys in the object.
-
Declared Space (e.g.,
let x = 0
):- When space is explicitly declared, as in
let x = 0;
, it is Constant space O(1) because the amount of space required is fixed.
- When space is explicitly declared, as in
Space complexity analysis is crucial for understanding memory usage as input sizes increase.
Notation | Description | Example |
---|---|---|
O(1) | Constant time | Accessing an element in an array by index |
O(log n) | Logarithmic time | Binary search in a sorted array |
O(n) | Linear time | Finding the maximum element in an unsorted array |
O(n log n) | Linearithmic time | Merge sort, heap sort |
O(n^2) | Quadratic time | Bubble sort, insertion sort |
O(2^n) | Exponential time | Recursive solutions with branching |
O(n!) | Factorial time | Permutations of a set |
O(1)
function add(num1, num2, num3) {
return num1 + num2 + num3;
}
function sayHello() {
for (let i = 0; i < 100; i++) {
console.log("Hello");
}
}
function logMultiples(num) {
for (let i = 0; i < 10; i++) {
console.log(i * num);
}
}
O(n)
function sayHello(numberOfTimes) {
for (let i = 0; i < numberOfTimes; i++) {
console.log("Hello");
}
}
function doubleThenTriple(numbers) {
let doubled = numbers.map(function (num) {
return num * 2;
});
return doubled.map(function (num) {
return num * 3;
});
}
O(n2)
function allPairs(arr) {
let pairs = [];
for (let i = 0; i < arr.length; i++) {
for (let j = i + 1; j < arr.length; j++) {
pairs.push([arr[i], arr[j]]);
}
}
return pairs;
}
function bubbleSort(arr) {
let len = arr.length;
let lastSwap;
let temp;
while (len != 0) {
lastSwap = 0;
for (let i = 1; i < len; i++) {
if (arr[i - 1] > arr[i]) {
// Swap the two elements
temp = arr[i - 1];
arr[i - 1] = arr[i];
arr[i] = temp;
lastSwap = i;
}
}
len = lastSwap;
}
}
It’s a helpful rule of thumb that in general, if you see nested loops, the runtime will be O(nlevels of nesting). In other words, a function with a single for loop will be O(n), a function with a loop inside of a loop will be O(n2), a function with a loop inside of a loop inside of a loop will be O(n3), and so on. However, this rule of thumb doesn’t always hold, as the following examples show:
function logMultiples(n) {
for (let num1 = 1; num1 <= n; num1++) {
for (let num2 = 1; num2 <= n; num2++) {
console.log(num1 * num2);
}
}
}
function logSomeMultiples(n) {
for (let num1 = 1; num1 < n=; num1++) {
for (let num2 = 1; num2 <= Math.min(n, 10); num2++) {
console.log(num1 * num2);
}
}
}
The first function, logMultiples, is O(n2). When n is 1, the function logs 1 multiple; when n is 2, the function logs 4 multiples; when n is 3, the function logs 9 multiples, and so on.
What about the second function, logSomeMultiples? It might seem like this function should be O(n2) as well, because of the inner loop. However, the runtime of the inner loop is not proportional to n! The inner loop will run a maximum of 10 times, so it is actually an O(1) operation. This means that the outer loop, which is still O(n), performs an O(1) operation for each value of num1. Therefore, logSomeMultiples is itself O(n), not O(n2).