JavaScript Manual

JavaScript Data Structure

· Prabesh Thapa

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 specified start to end where end 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 integer
    • Uint8Array - 8-bit unsigned integer
    • Uint8ClampedArray: 8-bit unsigned integer
    • Int16Array: 16-bit two’s complement signed array
    • Uint16Array: 16-bit unsigned array
    • Int32Array: 32-bit two’s complement unsigned array
    • Uint32Array : 32-bit unsigned array
    • Float32Array: 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 integer
    • BigUint64Array: 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.