JavaScript Data Structure
Table of Content
Data Structure in JavaScript
Groups of data in different forms are one of the fundamental data structures in most of the programming languages. Normally, groups of data expressed through different data types are known as Collection
.
The three main Collection groups in JavaScript are -
- Indexed Collections
- Keyed Collections
- DOM Collections
Indexed Collection
Indexed collection is the colletion of data which is listed by their index. JavaScript collection indicies are 0-based, meaning they start at index 0 and go up to n-1
, where n
is the number of objects in the collection. JavaScript offers two kinds of indexed collections - Arrays
and Typed Arrays
.
Array Object
JavaScript Arrays are dynamically created special kind of object. Arrays store a numbers of variables including zero, which are known as empty array. These variables do not necesarily need to have same type. The variables stored in array have no names. These variables are accessed using non-negative integer index value.
If an array has n
variables, we can say that the length of that array is n
. These variables are referenced using integer index form 0
to n-1
.
- Arrays are resizable and can contain mixture of different data types.
- Arrays are zero-indexed i.e. the first element of array is at index
0
. - Copying arrays creat e a shallow copy, rather than deep copies.
A shallow copy of an object is a copy whose properties share the same references (point to the same underlying values) as those of the source object from which the copy was made. As a result, when you change either the source or the copy, you may also cause the other object to change too — and so, you may end up unintentionally causing changes to the source or copy that you don’t expect [mdn]
.
A deep copy of an object is a copy whose properties do not share the same references (point to the same underlying values) as those of the source object from which the copy was made. As a result, when you change either the source or the copy, you can be assured you’re not causing the other object to change too; that is, you won’t unintentionally be causing changes to the source or copy that you don’t expect[mdn]
.
Properties of an Array
Array object has one property, length
. the length
property indicates the number of component stores in that array.
Synatx:
arrayObject.length;
Examples:
const arr1 = [1, 2, 3];
const arr2 = new Array(9);
const arr3 = new Array(-10); // Negative numbers are not allowed
console.log(arr1.length); // OUTPUT: 3
console.log(arr2.length); //OUTPUT: 9
console.log(arr3.length); //OUTPUT: RangeError: Invalid array length
Adding element to an Array
We can add an element to an Array by accessing an element through it’s index and assign a value to it.
let friends = new Array(5);
friends[0] = "Suman";
friends[1] = "Anil";
friends[2] = "Yogendra";
friends[3] = "Bipin";
friends[4] = "Shusan";
console.log(friends);
//OUTPUT: ["Suman", "Anil", "Yogendra", "Bipin", "Shusan"]
In the above example, we have allocated 5 spaces for the array but we still can shrink or expand the size at any tine since the size ais dynamically allocated.
friends[5] = "Laxmi";
friends[6] = "Sumi";
console.log(friends);
// OUTPUT: ["Suman", "Anil", "Yogendra", "Bipin", "Shusan", "Laxmi", "Sumi"]
Array Methods
Here, we will have a look at the nmost commonly used methods:
push()
- Adds an element at the end of an array
let even = [2, 4, 6, 8];
even.push(10);
console.log(even); //OUTPUT: [2,4,6,8,10]
pop()
- removes the last element of an array
let even = [2, 4, 6, 8];
even.pop();
console.log(even); //OUTPUT: [2,5,6]
concat()
- joins arrays (two or more) into a single array
//Concat 2 Arrays
let even = [2, 4, 6, 8];
let odd = [1, 3, 5, 7, 9];
let array1 = odd.concat(even);
console.log(array1);
//OUTPUT: [1,3,5,7,2,4,6,8];
//Concat 3 Arrays
let doubles = [10, 11, 12, 13];
let array2 = even.concat(odd, doubles);
console.log(array2);
//OUTPUT: [2,4,6,8,1,3,5,7,9,10,11,12,13]
join()
- creates and returns a new string by conctenating all the elements in an array seperated by commas(,)
or user-specified seperator.
const life = ["Live", "Laugh", "Love"];
console.log(life.join());
// OUTPUT: "Live,Laugh,Love"
console.log(life.join(""));
// OUTPUT: "LiveLaughLove"
console.log(life.join("-"));
// OUTPUT: "Live-Laugh-Love"
reverse()
- Just as the name suggests, it reverses the order of elements in an array.
let even = [2, 4, 6, 8, 10];
let reversed = even.reverse();
console.log(reversed);
// OUTPUT: [10,8,6,4,2]
slice(start, end)
- It returns a shallow copy of portion of an array into an new array from specifiedstart
toend
whereend
is not included. The original array is not modified.
const hobbies = ["cricket", "football", "anime", "bikinig", "hiking"];
console.log(hobbies.slice(2));
//OUTPUT: ["anime", "bikinig", "hiking"]
console.log(hobbies.slice(2, 4));
//OUTPUT: ["anime", "bikinig"]
console.log(hobbies.slice(1, 5));
//OUTPUT: ["football", "anime", "bikinig", "hiking"]
console.log(hobbies.slice(-3));
//OUTPUT: ["anime", "bikinig", "hiking"]
console.log(hobbies.slice(2, -1));
//OUTPUT: ["anime", "bikinig"]
console.log(hobbies.slice());
//OUTPUT: ["cricket", "football", "anime", "bikinig", "hiking"]
TypedArray Object
Array
object can be used to store different types of elements in one array and also offers various powerful methods to manipulate the elements in the array.
However, when we need to work with raw binary data, we need to use the TypedArray
Object. Raw data is processed when manipulating audio and video for example.
JavaScript typed arrays are array-like objects that provide a mechanism for reading and writing raw binary data in memory buffers. JavaScript typed arrays are divided into Buffers
and Views
.
- Buffer - An object representing a chunk of data implemented by the
ArrayBuffer
object. It is used to represent a fixed-length binary data buffer. To represent this buffer, we create a view -DataView
and use it to read and write the contents of the buffer. There are various types of views which represent the most common numeric types -Int8Array
: 8-bit two’s complement signed integerUint8Array
- 8-bit unsigned integerUint8ClampedArray
: 8-bit unsigned integerInt16Array
: 16-bit two’s complement signed arrayUint16Array
: 16-bit unsigned arrayInt32Array
: 32-bit two’s complement unsigned arrayUint32Array
: 32-bit unsigned arrayFloat32Array
: 32-bit IEEE floating point number (7 significant digits)Float64Array
: 64-bit IEEE floating point number (16-bit significant digits)BigInt64Array
: 64-bit two’s complement signed integerBigUint64Array
: 64-bit unsigned integer
Keyed Collections
A keyed collection is a collection of data represented in key-value notation. These values are acccessed and manipulated with their respective keys. We have two kinds of keyed collection in JavaScript: Map
and Set
.
Map
Map object is the collection of key-value pairs. Keys are always unique in Map meaning, a key in a Map object can only appear once.
Creating a Map Object
Suppose, we have a list of user
object as below:
let virat = { fullName: "Virat Kohli" },
smith = { fullName: "Steve Smith" },
root = { fullName: "Joe Root" };
Assuming we have to create a map of these players and their country, we can code as below:
let fabThree = new Map();
console.log(typeof fabThree); // OUTPUT: object
console.log(fabThree instanceof Map); // OUTPUT: true
The fabThree
is an instance of Map
object and it’s type is an object as shown above.
Adding elements to Map
To assign country to these players, we can use the set()
method.
fabThree.set(virat, "India").set(smith, "Australia").set(root, "England");
The set()
method maps each players with their specified country.
Initialize a map with an iterable object
We can pass an iterable object to the Map()
constructor.
Syntax:
let map = new Map([iterable]);
Code:
let fabThree = new Map([
[kohli, "India"],
[smith, "Australia"],
[root, "England"],
]);
Getting an element from a map by Key
We can use the get()
method to get an element from a map with key.
fabThree.get(smith); //OUTPUT: Australia
If we somehow pass a key that does not exist, the get()
method will return undefined.
fabThree.get(foo); //OUTPUT: undefined
Checking the existence of an element by Key
We can use the has()
method to check if the key exists in the map.
fabThree.has(bar); //OUTPUT: false
fabThree.has(root); //OUTPUT: true
Get the numbers of elements in the map
We can get the numbers of elements we have in the map with size
property. The size
property returns the number of elements in the Map object.
console.log(fabThree.size); //OUTPUT: 3
Iterate over map keys
We can use the keys()
method to obtain the keys of the Map
object. The keys()
method returns a new iterator object that contains the keys of the elements in the map (return value - array).
let virat = { fullName: "Virat Kohli" },
smith = { fullName: "Steve Smith" },
root = { fullName: "Joe Root" };
let fabThree = new Map([
[virat, "India"],
[smith, "Australia"],
[root, "England"],
]);
for (const player of fabThree.keys()) {
console.log(player.fullName);
}
OUTPUT:
Virat Kohli
Steve Smith
Joe Root
Iterate over Map values
We can also iterate over values like with keys using values()
method.
let virat = { fullName: "Virat Kohli" },
smith = { fullName: "Steve Smith" },
root = { fullName: "Joe Root" };
let fabThree = new Map([
[virat, "India"],
[smith, "Australia"],
[root, "England"],
]);
for (const player of fabThree.values()) {
console.log(player);
}
OUTPUT:
India
Australia
England
Iterate over entire map elements
We van use the entries()
method to iterate over map elements. The entries()
method returns an iteratorobject that contains array of [key, value]
of each element in that Map object.
let virat = { fullName: "Virat Kohli" },
smith = { fullName: "Steve Smith" },
root = { fullName: "Joe Root" };
let fabThree = new Map([
[virat, "India"],
[smith, "Australia"],
[root, "England"],
]);
for (let [player, country] of fabThree.entries()) {
console.log(`${player.fullName}: ${country});
}
OUTPUT:
Virat Kohli: India
Steve Smith: Australia
Joe Root: England
We can also use forEach()
instead of for...of
loop.
fabThree.foreach((player, country) => {
console.log(`${player.fullName}: ${country}`);
});
Delete an element by Key
We can use the delete()
method to delete an entry in the map.
fabThree.delete(root);
Delete all elements in the map
We can clear all the entries in the Map object with the clear()
method.
fabThree.clear();
Weak Map
Weak Map
is similar to Map
, the differnece being the keys of a WeakMap
must be an object. This means that when a reference to a key i.e. the object is out of scope, the corresponding value is automatically released from the memory.
The main differences between Map
and WeakMap
can be summarised as below -
- Elements of WeakMap cannot be iterated.
- We cannot clear all elements of WeakMap at once.
- We cannot check the size of a WeakMap.
Set
Set
is used to store a collection of unique values of any type.
Syntax:
let set = new Set();
The Set
constructore also accepts optional iterable object. If we pass an iterable object to the Set
constructor, all the elements of the iterable object will be aded to the new Set.
Creating a new Set from an Array
let marks = new Set([9, 5, 6, 4, 5, 6, 9, 4, 5]);
Here, marks array contains only 4 distict elements. All of the elements in the Set
must be unique, therefore it returns a object with these distinct elements.
console.log(marks);
OUTPUT:
Set {9, 5, 6, 4}
console.log(typeof marks);
//OUTPUT: Object
The marks
set is an instance of the Set
type.
console.log(marks instanceof Set);
//OUPTUT: true
We can get the size of the Set
object with the size
property.
console.log(marks.size);
//OUTPUT: 4
Adding an elements to a Set
We can add an element to a Set with the add()
method.
marks.add("2");
console.log(marks); //OUTPUT: Set {9,5,6,4,2}
The add()
method is chainable, so we can add multiple elements to the Set at once.
marks.add(3).add(10).add(1);
Checking if a value exists in the Set
We can check if certain elements exists in the Set with has()
method. This returns true
if the elements exists and false
otherwise.
console.log(marks.has(9)); //OUTPUT: true
console.log(marks.has(100)); //OUTPUT: false
Removing elements from a Set
We can delete a speicified elements from the Set with delete()
method. the delete()
method returns true which indicates that the element has been deleted.
marks.delete(6);
console.log(marks); //OUTPUT: Set{9,5,4,2}
We can also delete entire elements in the Set with the clear()
method.
marks.clear();
console.log(marks); //OUTPUT: Set{}
Set
also provides the keys()
, values()
and entries()
methods just like in Map
. However, both keys and values are same in Set
.