Understanding JavaScript Concepts: A Comprehensive Guide

Understanding JavaScript Concepts: A Comprehensive Guide

JavaScript is a versatile and widely-used programming language that powers dynamic web applications and adds interactivity to websites. As developers dive into the world of JavaScript, they encounter various concepts and terminologies that are essential to grasp in order to write efficient and robust code. In this article, we will explore a comprehensive list of JavaScript concepts, from execution context and closures to event bubbling and shadowing. By understanding these fundamental concepts, you will gain valuable insights into how JavaScript works and be better equipped to tackle complex programming challenges. So, let's embark on a journey to unravel the intricacies of JavaScript and build a solid foundation for your coding endeavors.

Table of content:

  1. Closures
  2. Deep Comparison
  3. Deep Copying
  4. Event Bubbling
  5. Event Capturing
  6. Event Loop
  7. Execution context
  8. Global context
  9. Hoisting
  10. Immediately Invoked Function Expressions (IIFE)
  11. Microtask Queue
  12. Shadowing
  13. Shallow Comparison
  14. Shallow Copying
  15. Temporal Dead Zone (TDZ)


Closures:

Closures are an important concept in JavaScript that allow functions to retain access to variables from the parent scope, even after the parent function has finished executing. This is achieved by creating a reference to the variables in memory, which is known as a closure.

Here's an example to help illustrate closures:

function outerFunction() {
  var outerVariable = 'I am from the outer function!';


  function innerFunction() {
    console.log(outerVariable);
  }


  return innerFunction;
}


var closureExample = outerFunction();
closureExample(); // Output: I am from the outer function!

        

In this example, outerFunction defines a local variable called outerVariable and also defines an inner function called innerFunction. Inside innerFunction, we try to access outerVariable, which is outside of its own scope.

When outerFunction is called and assigned to the variable closureExample, it returns the innerFunction itself. At this point, the outerFunction has finished executing, and ordinarily, we might expect the outerVariable to be no longer accessible.

However, since innerFunction is a closure, it still retains a reference to its parent scope and can access outerVariable even after outerFunction has completed. When we invoke closureExample(), it logs the value of outerVariable to the console.

This is because the closure has "closed over" the outerVariable and preserved its state. Closures are powerful because they allow us to create functions with persistent access to variables that would otherwise be inaccessible.

Here's another example that showcases the concept of closures with a counter:

function counter() {
  var count = 0;


  function increment() {
    count++;
    console.log(count);
  }


  function decrement() {
    count--;
    console.log(count);
  }


  return {
    increment: increment,
    decrement: decrement
  };
}


var counterInstance = counter();
counterInstance.increment(); // Output: 1
counterInstance.increment(); // Output: 2
counterInstance.decrement(); // Output: 1

        

In this example, the counter function returns an object with two methods: increment and decrement. Inside counter, there's a local variable count that is shared by both methods.

When we call counter and assign it to counterInstance, we essentially create a closure for the count variable. Each time we call increment or decrement, the count variable is updated and its value is printed to the console.

Closures are a powerful feature in JavaScript and are commonly used for encapsulation, data privacy, and creating modular code. They provide a way to maintain state and retain access to variables within the scope of a function, even after it has finished executing.


Deep Comparison:

The deep comparison involves comparing the values and contents of objects or arrays recursively. Instead of comparing the references, a deep comparison checks whether the structures and values of the objects or arrays are equivalent.

Example:

const obj1 = { name: 'John', age: 25 };
const obj2 = { name: 'John', age: 25 };
const obj3 = { name: 'Jane', age: 30 };


console.log(obj1 === obj2); // Output: false (different memory references)
console.log(obj1.name === obj2.name && obj1.age === obj2.age); // Output: true (deep comparison)


console.log(deepCompare(obj1, obj2)); // Output: true (custom deep comparison function)
console.log(deepCompare(obj1, obj3)); // Output: false (custom deep comparison function)        

In this example, obj1 and obj2 have the same properties and values, but they are distinct objects in memory. The comparison obj1 === obj2 returns false. However, a deep comparison based on property values (obj1.name === obj2.name && obj1.age === obj2.age) results in true.

Deep comparison often requires custom logic or helper functions to recursively compare nested properties or elements. It compares the values and structures of objects or arrays rather than their references.

Here's a simplified example of a deep comparison function:

function deepCompare(a, b) {
  // Compare primitive values
  if (a === b) {
    return true;
  }


  // Compare object or array properties recursively
  if (typeof a === 'object' && typeof b === 'object') {
    const keysA = Object.keys(a);
    const keysB = Object.keys(b);


    if (keysA.length !== keysB.length) {
      return false;
    }


    for (let key of keysA) {
      if (!deepCompare(a[key], b[key])) {
        return false;
      }
    }


    return true;
  }


  return false;
}        

This deepCompare function recursively compares the properties of objects or arrays. It returns true if the structures and values of the objects or arrays are equivalent.


Deep Copying:

Deep copying creates a completely independent and separate copy of an object or array, including all nested objects/arrays. It means that the new copy has its own set of values, and modifying the copied object/array or any of its nested objects/arrays does not affect the original.

// Deep copying an array
const originalArray = [1, 2, [3, 4]];
const deepCopyArray = JSON.parse(JSON.stringify(originalArray));


console.log(originalArray === deepCopyArray); // Output: false (different references)


console.log(originalArray[2] === deepCopyArray[2]); // Output: false (different references)        

In the example above, JSON.parse(JSON.stringify()) is used to deep copy the originalArray. It converts the array to a JSON string and then parses it back into a new array. This process creates a new copy with completely separate references for both the top-level and nested arrays.

Deep copying objects can be done similarly using JSON.parse(JSON.stringify()) or through libraries such as Lodash with its cloneDeep() method.

Deep copying guarantees that the copied object/array and its nested objects/arrays are independent. However, it can be more resource-intensive and might not handle certain complex objects with non-enumerable properties or functions.


Event Bubbling:

Event bubbling refers to the propagation of an event from the innermost element to the outermost element in the DOM hierarchy. In other words, when an event occurs on a child element, it "bubbles up" to its parent elements until it reaches the top-level element (usually the document object).

Here is an example code snippet that demonstrates event bubbling in action:

<body>
  <div id="outer">
    <div id="inner">
      <button id="myButton">Click me!</button>
    </div>
  </div>
</body>

const myButton = document.getElementById('myButton')
const innerDiv = document.getElementById('inner');
const outerDiv = document.getElementById('outer');


myButton.addEventListener('click', () => {
  console.log('Button clicked!');
});


innerDiv.addEventListener('click', () => {
  console.log('Inner div clicked!');
});


outerDiv.addEventListener('click', () => {
  console.log('Outer div clicked!');
});        

In this example, if you click on the button, you will see three log messages in the console: "Button clicked!", "Inner div clicked!", and "Outer div clicked!". This is because the click event "bubbles up" from the button to the inner div, then to the outer div, and finally to the document object.


Event Capturing:

Event capturing is the opposite of event bubbling. It refers to the propagation of an event from the top-level element to the innermost element in the DOM hierarchy. In other words, when an event occurs on a parent element, it "captures" the event before it reaches its child elements.

<body>
  <div id="outer">
    <div id="inner">
      <button id="myButton">Click me!</button>
    </div>
  </div>
</body>

const myButton = document.getElementById('myButton');
const innerDiv = document.getElementById('inner');
const outerDiv = document.getElementById('outer');


myButton.addEventListener('click', () => {
  console.log('Button clicked!');
}, false);


innerDiv.addEventListener('click', () => {
  console.log('Inner div clicked!');
}, false);


outerDiv.addEventListener('click', () => {
  console.log('Outer div clicked!');
}, true);        

In this example, if you click on the button, you will see two log messages in the console: "Outer div clicked!" and "Inner div clicked!". This is because the click event is first captured by the outer div (since its event listener is set to true for event capturing), then it propagates down to the inner div, and finally to the button. The button's event listener is set to false, which means it uses event bubbling instead of event capturing. Therefore, its event handler is only called after the capturing phase is complete.


Event Loop:

To understand the event loop in JavaScript, let's break down the different components involved: the heap, call stack, callback queue, and message queue.

  • Heap:

The heap is a region in memory where objects and variables are stored. It is where dynamically allocated memory resides, including objects, function closures, and other data.

  • Call Stack:

The call stack is a data structure that keeps track of the currently executing functions. It records the execution context of each function, including local variables, function arguments, and the point of execution. When a function is called, it is pushed onto the call stack, and when a function completes, it is popped off the stack.

  • Web APIs and Event Listeners:

In a browser environment, JavaScript has access to Web APIs provided by the browser, such as the DOM API, AJAX API, and setTimeout. These APIs are asynchronous and run in the background without blocking the main thread. When an asynchronous operation is initiated, such as making an AJAX request or setting a timer with setTimeout, it is passed to the relevant Web API for execution.

Event listeners are functions that wait for specific events (e.g., click, load, timer) to occur. When an event is triggered, the associated event listener function is added to the callback queue.

  • Callback Queue:

The callback queue (also known as the task queue or message queue) is a queue data structure that holds callback functions. These callbacks are created as a result of asynchronous operations or event listeners. When an asynchronous operation completes or an event is triggered, its associated callback is added to the callback queue.

  • Message Queue:

The message queue holds messages generated by the browser, such as rendering-related messages or user input events. These messages are processed by the event loop and are added to the message queue.

Now, let's see how these components interact in the event loop:

  1. JavaScript code is executed synchronously from top to bottom. Function calls are added to the call stack, and their execution begins.
  2. If an asynchronous operation (e.g., setTimeout, AJAX request) is encountered, it is handed off to the relevant Web API, and the callback related to that operation is registered.
  3. While the asynchronous operation is being handled by the Web API, the JavaScript engine continues executing the remaining code without waiting for the operation to complete. Other functions may be called and added to the call stack.
  4. Once the asynchronous operation finishes, the callback associated with it is placed in the callback queue.
  5. The event loop continuously monitors the call stack. If it is empty, the event loop checks the callback queue.
  6. If the callback queue is not empty, the event loop takes the first callback from the queue and pushes it onto the call stack, preparing it for execution.
  7. The callback function is executed, and any functions called within it are added to the call stack for execution.
  8. This process continues as long as there are callbacks in the callback queue.
  9. Additionally, while the event loop is running, it also checks the message queue for any messages. If a message is found, it is processed by the event loop, which may result in adding callbacks to the callback queue.

To summarize, the event loop constantly checks the call stack and callback queue. When the call stack is empty, it takes the next callback from the callback queue, pushes it onto the call stack, and executes it. This mechanism allows JavaScript to handle asynchronous operations and events efficiently while ensuring the overall order and execution of code.


Execution context:

Execution context is an internal JavaScript construct that holds information about the environment in which code is executed. It includes variables, function declarations, and the scope chain. Whenever a function is invoked or a script starts running, a new execution context is created.

An execution context consists of three main components:

  • Variable Object (VO)/Environment Record:

The variable object (VO) or environment record contains all the variables, function declarations, and function parameters defined within the scope of the execution context. It is a representation of the current scope's variables and functions.

  • Scope Chain:

The scope chain is a list of variable objects that represents the nested structure of the execution contexts. It allows for the resolution of variables and functions based on lexical scope. When a variable is accessed, JavaScript traverses the scope chain until it finds the variable or reaches the global object.

  • 'this' Value:

The this value refers to the object that is currently executing the code. It provides a reference to the context within which a function is called.

The execution context is created in two phases: the creation phase and the execution phase.

During the creation phase, the following steps occur:

  • Creation of the Variable Object:

The variable object (VO) is created, and all function and variable declarations are hoisted. This means that the functions and variables are registered in memory but not assigned any values yet. Functions can be invoked before their actual declaration due to hoisting.

  • Creation of the Scope Chain:

The scope chain is created by linking the variable objects of the current and parent execution contexts. This establishes the lexical scope for variable resolution.

  • Determining the 'this' Value:

The this value is set based on how the function is called. It can refer to the global object (in non-strict mode) or the object calling the function (in methods or constructors).

During the execution phase, the code is executed line by line, variables are assigned values, functions are invoked, and expressions are evaluated.

Understanding the concept of execution contexts helps in comprehending variable scoping, function invocation, and the resolution of identifiers in JavaScript. It plays a crucial role in how code is executed and how variables and functions are accessed within different scopes.


Global context:

Global context refers to the environment in which the top-level code of a script is executed. It represents the outermost scope of a JavaScript program, and any variables or functions defined in this context are accessible throughout the entire program.

When a JavaScript program starts running, a global object is created. In a browser environment, this global object is referred to as the window object. In Node.js, it is referred to as the global object.

The global context has a global scope, meaning that variables and functions declared in the global context are accessible from any other scope within the program. These variables and functions are often referred to as global variables and global functions.

Here are a few key points about the global context:

  • Variable Declarations: When you declare a variable outside of any function or block in a JavaScript program, it becomes a global variable and is added as a property of the global object.

var globalVariable = 'Hello, world!';
console.log(globalVariable); // Output: Hello, world!
console.log(window.globalVariable); // Output: Hello, world! (in a browser environment)        

  • Function Declarations: Functions defined in the global context become global functions that can be accessed from anywhere in the program.

function globalFunction() {
  console.log('This is a global function');
}


globalFunction(); // Output: This is a global function
window.globalFunction(); // Output: This is a global function (in a browser environment)        

  • Execution Context: The global context has its own execution context, which includes a global scope, a global object, and a this reference that points to the global object.
  • Accessibility: Variables and functions declared in the global context can be accessed from other scopes within the program, such as functions or blocks.

function foo() {
  console.log(globalVariable); // Accessible within other scopes
}        

  • Avoiding Global Pollution: It's generally recommended to minimize the use of global variables and functions to prevent naming conflicts and maintain code modularity. Instead, encapsulate your code within functions or modules to create local scopes.

(function () {
  var localVariable = 'Hello, local!';
  console.log(localVariable); // Output: Hello, local!
})();


console.log(typeof localVariable); // Output: undefined (outside the function scope)        

Understanding the global context is essential for writing modular and maintainable JavaScript code. It's important to be mindful of the variables and functions you define in the global context to avoid unintended consequences and potential conflicts with other parts of your program.


Hoisting:

Hoisting is a JavaScript mechanism where variable and function declarations are moved to the top of their scope before the code is executed. This means that you can use a variable or function before it's declared.

Here's an example of variable hoisting:

console.log(x); // undefined 
var x = 5;         

Even though we're trying to log the value of x before it's declared, the code runs without an error and logs undefined. This is because the var declaration for x is hoisted to the top of its scope, so when we try to log x, it exists but has no value yet.

Here's an example of function hoisting:

foo(); // "Hello, world!" 

function foo() { 
 console.log("Hello, world!"); 
}         

In this case, we're calling the foo() function before it's declared. However, the code runs without an error and logs "Hello, world!". This is because function declarations are also hoisted to the top of their scope.

It's important to note that only variable and function declarations are hoisted, not variable assignments or function expressions. Here's an example to illustrate this:

console.log(y); // Uncaught ReferenceError: y is not defined y = 5;         

In this case, we're trying to log the value of y before it's declared or assigned. However, we get a ReferenceError because there's no hoisting for variable assignments.

Overall, hoisting can be a useful feature of JavaScript, but it can also lead to unexpected behavior if you're not careful. It's important to understand how hoisting works in order to write clean and maintainable code.


Immediately Invoked Function Expressions (IIFE):

Immediately Invoked Function Expressions (IIFE) are JavaScript functions that are executed as soon as they are defined. They are often used to create a private scope and prevent polluting the global scope with variables and functions.

The syntax for an IIFE involves wrapping the function declaration or expression within parentheses, followed by an immediate invocation using parentheses or brackets.

Here's an example of an IIFE:

(function() {
  // Code to be executed immediately
})();        

In this example, an anonymous function is declared inside parentheses. The final pair of parentheses () immediately invokes the function.

IIFEs can also accept arguments:

(function(message) {
  console.log(message);
})('Hello, IIFE!');        

In this case, the argument 'Hello, IIFE!' is passed to the IIFE, and it is immediately invoked with the given argument.

Benefits and Use Cases of IIFEs:

  1. Encapsulation: IIFEs create a private scope, allowing you to declare variables and functions that are not accessible outside the function. This helps prevent naming conflicts and keeps your code modular.
  2. Avoiding Global Pollution: By encapsulating code within an IIFE, you can avoid adding unnecessary variables and functions to the global scope, thus reducing the chance of naming collisions with other parts of your program.
  3. Immediate Execution: IIFEs allow you to execute code immediately without the need for external triggers. This can be useful for initializing code or performing one-time setup operations.
  4. Preserving Variable Privacy: Variables declared within the IIFE are not accessible from the outside, providing a way to create private variables and functions.

Here's an example that demonstrates variable privacy using an IIFE:

var counter = (function() {
  var count = 0;


  return {
    increment: function() {
      count++;
      console.log(count);
    },
    decrement: function() {
      count--;
      console.log(count);
    }
  };
})();


counter.increment(); // Output: 1
counter.increment(); // Output: 2
counter.decrement(); // Output: 1        

In this example, the counter variable is assigned the result of the IIFE. The IIFE creates a private count variable that is inaccessible from the outside but can be modified and accessed through the returned object's increment() and decrement() methods.

Overall, IIFEs are a powerful tool for creating encapsulated code and managing the scope of variables and functions in JavaScript. They provide a way to execute code immediately while preserving privacy and avoiding global pollution.


Microtask Queue:

The Microtask Queue, also known as the Promise Job Queue or the Job Queue, is a concept in JavaScript that handles the execution of microtasks. Microtasks are tasks that need to be executed as soon as possible, typically after the currently executing task has finished.

In the event loop, the Microtask Queue is a separate queue that is processed after the Call Stack is empty, but before rendering and user input events. It has a higher priority than the regular callback queue (also known as the task queue or message queue).

Microtasks are primarily associated with Promises and certain APIs like process.nextTick() in Node.js or queueMicrotask() in modern browsers. When a Promise resolve or rejects, or when a microtask API is called, the corresponding microtask is added to the Microtask Queue.

The behavior of the Microtask Queue can be summarized as follows:

  1. Task Execution: When a task (regular script execution, event callback, or timer callback) completes, the event loop checks if the Call Stack is empty.
  2. Microtask Queue Processing: If the Call Stack is empty, the event loop processes all the microtasks in the Microtask Queue, one by one, until the queue is empty.
  3. Microtask Execution: Each microtask is taken from the Microtask Queue and executed. This continues until the queue is empty or until a maximum number of iterations has been reached (to prevent infinite loops).
  4. User Interface Rendering: After the Microtask Queue is empty, the event loop proceeds to render UI updates and handle user input events.

By using the Microtask Queue, JavaScript provides a way to schedule microtasks for immediate execution after the current task. This helps in maintaining a responsive and efficient application by allowing time-critical tasks to be handled quickly.

Here's an example that demonstrates the order of execution between the Microtask Queue and the regular callback queue:

console.log('Start');


setTimeout(() => {
  console.log('setTimeout');
}, 0);


Promise.resolve().then(() => {
  console.log('Promise');
});


console.log('End');        

In this example, the output would be:

Start
End
Promise
setTimeout        

Even though setTimeout() is scheduled with a delay of 0 milliseconds, it is added to the regular callback queue. The microtask Promise.resolve().then() is added to the Microtask Queue and executed before the setTimeout() callback.

It's important to note that microtasks should be kept short and efficient, as they can potentially block the event loop if they are not yielding control back to the browser or the JavaScript runtime. Excessive or lengthy microtasks can result in unresponsive behavior, impacting user experience.

Overall, the Microtask Queue provides a mechanism to schedule and execute microtasks in a timely manner, ensuring the order and responsiveness of JavaScript execution.


Shadowing:

Variable shadowing occurs when a variable declared within a block of code (e.g., a function) has the same name as a variable declared in an outer scope. The inner variable "shadows" the outer variable, effectively hiding it from the inner scope.

Here's an example of variable shadowing in JavaScript:


let x = 10; // outer variable 
function myFunction() { 
  let x = 20; // inner variable, shadows outer variable 
  console.log(x); // prints 20 
} 
myFunction(); 
console.log(x); // prints 10         

In this example, we declare a variable x with a value of 10 in the outer scope. We then declare a function myFunction that declares a variable x with a value of 20 in its inner scope. When we call myFunction, it prints the value of x within its inner scope, which is 20. However, when we call console.log(x) outside of myFunction, it prints the value of the outer variable x, which is still 10.

Variable shadowing can be confusing and can lead to bugs in your code, especially if you're not aware of which variables are being used in which scope. It's generally a good practice to avoid variable shadowing whenever possible and to use unique variable names to avoid naming conflicts. There are cases where shadowing is illegal and cases where it is legal. Let's explore both scenarios with code snippets:

  • Illegal Shadowing:

Illegal shadowing occurs when a variable declared in an inner scope conflicts with a variable in an outer scope and the inner variable is of the same type (e.g., both are variables or both are functions). This can lead to unexpected behavior and should be avoided.

var x = 5;


function outer() {
  var x = 10; // Illegal shadowing of 'x' variable in the outer scope
  console.log(x); // Output: 10
}


outer();
console.log(x); // Output: 5 (outer 'x' variable is unaffected)

        

In the above example, the inner x variable shadows the outer x variable within the outer function. This means that when x is referenced within the function, it refers to the inner x variable, not the outer one. The outer x variable remains unchanged outside the function.

  • Legal Shadowing:

Legal shadowing occurs when a variable declared in an inner scope has the same name as a variable in an outer scope, but they serve different purposes or have different types. This can be intentional and can help to clarify code or encapsulate logic.

var x = 5;


function outer() {
  var x = "inner"; // Legal shadowing of 'x' variable in the outer scope with a different type
  console.log(x); // Output: "inner"
}


outer();
console.log(x); // Output: 5 (outer 'x' variable is unaffected)

        

In this example, the inner x variable is a string, while the outer x variable is a number. Since they have different types, they can coexist without conflict. The inner x variable within the outer function shadows the outer x variable, and each variable retains its own value and scope.

It's generally recommended to avoid shadowing variables as it can introduce confusion and make the code harder to maintain. However, legal shadowing can be used judiciously for specific purposes, such as encapsulating logic within a function or introducing temporary variables.


Shallow Comparison:

The shallow comparison involves comparing the references of objects or arrays rather than their contents. When performing a shallow comparison, two objects or arrays are considered equal only if they reference the same memory location.

Example:

const arr1 = [1, 2, 3];
const arr2 = [1, 2, 3];
const arr3 = arr1;


console.log(arr1 === arr2); // Output: false (different memory references)
console.log(arr1 === arr3); // Output: true (same memory reference)        

In this example, arr1 and arr2 have the same elements, but they are stored in different memory locations. Hence, the comparison arr1 === arr2 returns false. On the other hand, arr1 and arr3 reference the same memory location, so the comparison arr1 === arr3 returns true.

Shallow comparison is the default behavior for most comparison operations in JavaScript, including the equality operator (== or ===).


Shallow Copying:

Shallow copying creates a new object or array and copies the references of the nested objects or arrays from the original object/array. In other words, it creates a new copy of the top-level object/array, but the nested objects/arrays are still referenced and shared between the original and copied objects/arrays.

// Shallow copying an array
const originalArray = [1, 2, [3, 4]];
const shallowCopyArray = [...originalArray];


console.log(originalArray === shallowCopyArray); // Output: false (different references)


console.log(originalArray[2] === shallowCopyArray[2]); // Output: true (same reference)        

In the example above, the originalArray is shallow copied using the spread operator (...). The resulting shallowCopyArray is a new array with a different reference. However, the nested array [3, 4] is still referenced by both arrays.

Shallow copying objects can be done using methods like Object.assign() or the spread operator ({...}).

Shallow copying is a lightweight approach and useful in many scenarios. However, it has limitations when you need to modify or manipulate nested objects/arrays without affecting the original object/array.


Temporal Dead Zone (TDZ):

Temporal Dead Zone (TDZ) is a behavior in JavaScript that occurs when you try to access a variable before it has been declared using the let or const keywords. When a variable is declared using let or const, it is hoisted to the top of its block scope, but it is not initialized until the line where it is declared is executed. Any attempt to access the variable before that line is executed will result in a ReferenceError.

Here's an example:

console.log(myVar); // ReferenceError: myVar is not defined 
let myVar = 'hello world';         

In this example, we are trying to access the myVar variable before it has been declared. This results in a ReferenceError because the variable is in the TDZ.

To avoid TDZ, you should always declare your variables before you use them. Here's the correct way to write the above code:

let myVar = 'hello world'; 
console.log(myVar); // outputs 'hello world'         

Now, the myVar variable is declared and initialized before we try to access it, so there is no TDZ and the code executes without errors.


#Closures #DeepComparison #DeepCopying #EventBubbling #EventCapturing #EventLoop #Executioncontext #Globalcontext #Hoisting #ImmediatelyInvokedFunctionExpressions(IIFE) #MicrotaskQueue #Shadowing #ShallowComparison #ShallowCopying #TemporalDeadZone(TDZ)

To view or add a comment, sign in

More articles by Abdulmoiz Ahmer

Insights from the community

Others also viewed

Explore topics