/javascript_enigmas

Repo that contains examples that tells about the odds of the javascript language. Output based javascript question with solutions

Javascript Enigmas - Odds of Javascript:

  • Contains output based questions for javascript.
  • Example explains the odds of javascript.

Question 1:

What will be the output for below javascript code?

var a = 10;
var b = '10';
console.log(a == b);
Click to view output

Output:

true.

Explanation:

== operator in javascript only compares the values and not the type hence it will log true.

Question 2:

What will be the output for below javascript code?

console.log('start');

const fn = () =>
  new Promise((resolve, reject) => {
    console.log(1);
    resolve('success');
  });

console.log('middle');

fn().then((res) => {
  console.log(res);
});

console.log('end');
Click to view output

Output:

start middle 1 end success

Explanation:

Since Javascript is a single threaded language, it can process only one statement at a time. Also we know that Promises are async code that will be kept aside in the task queue to be executed later until the main thread / synchronous code is executed.

Now you might think that fn() is executed then the Promise will be push the callback queue but here is a catch, Promise does not go to the callback queue, its the then() function which is a special kind of callback function that receives the resolved value.

So when we call fn() it creates a new Promise object and also takes a executor function and finds out that there is a console.log() which is synchronous code, so it will be executed right away and then resolve() will provide the value for it's then() method to use the value as response.

So, below are the order the output will be printed...

  1. Log "start"
  2. Function declared
  3. Log "middle"
  4. Calling function fn(): will log "1" and resolve "success" as a value which will be executed later once the main thread is executed.
  5. Log "end"
  6. Main thread is completed, now promise value will be printed "success"

Question 3:

What will be the output for below javascript code?

const a = [{ name: 'John' }, { name: 'Jane' }];

const b = [...a];

b[0] = { name: 'Oliver' };
b[1].age = 30;

console.log(a);
console.log(b);
Click to view output

Output:

a = [ { name: 'John' }, { name: 'Jane', age: 30 } ]

b = [ { name: 'Oliver' }, { name: 'Jane', age: 30 } ]

Explanation:

  1. const a = [{name: "John"}, {name: "Jane"}] - Creates a variable a with an array that contains 2 objects.
  2. const b = [...a]; - Creates a shallow copy of a, where reference of objects is being copied to variable b.
  3. b[0] = {name: "Oliver"} - Replacing the first object in variable b with a brand new object, without affecting the variable a.
  4. b[1].age = 30; - Now updating the second object inside variable b which is the reference of second object in variable a, will update the values in both a and b as they both are referencing to the same object.

Question 4:

What will be the output for below javascript code?

var x = 5;
var y = 2;
console.log(x ** y);
Click to view output

Output:

25

Explanation:

The ** operator is used for exponentiation in JavaScript. which is equal to Math.pow(5,2). ** is called exponential operator.

Question 5:

What will be the output for below javascript code?

function getTotal(...args) {
  console.log(typeof args);
}

getTotal(871);
Click to view output

Output:

object

Explanation:

When you use ...args as a parameter, it creates an array named args within the function that will hold all the arguments passed to the function. So, even though you call getTotal(871), the args array will still contain that single argument (871) as an array element. And as typeof Array is object.

Question 6:

What will be the output for below javascript code?

const a = [1, 2, 3];
const b = a;
a.push(4);

console.log(b.length);
Click to view output

Output:

4

Explanation:

  1. const a = [1, 2, 3]; creates a constant variable named a and assigns an array literal containing the numbers 1, 2, and 3 to it.
  2. const b = a; creates another constant variable named b and assigns the value of a to it. However, it's important to understand that this is not copying the array elements. Instead, it's creating a new reference (or alias) that points to the same array object in memory. Both a and b now refer to the same array.
  3. a.push(4); uses the push method to add the number 4 to the end of the array that a (and consequently b) refers to. Since a and b point to the same array, this modification affects both variables.
  4. console.log(b.length); logs the length of the array using b.length. Because a and b refer to the same modified array, this will print: 4

Question 7:

What will be the output for below javascript code?

const obj = { prop: 10 };
const arr = [obj];
arr[0].prop = 20;

console.log(obj.prop);
Click to view output

Output:

20

Explanation:

  1. const obj = { prop: 10 }; creates a constant object named obj with a single property named prop that has the value 10.
  2. const arr = [obj]; creates a constant array named arr and assigns a single element to it. That element is a reference to the obj object you created earlier. It's important to remember that you're not copying the object itself, but rather creating a new reference in the array that points to the same object in memory.
  3. arr[0].prop = 20; accesses the first element of the arr array (which is the reference to the obj object) and modifies the prop property within that object to have a value of 20. Because arr[0] and obj both refer to the same object, this change is reflected in both.
  4. console.log(obj.prop); logs the value of the prop property within the obj object. Since you modified the property in step 3, this will print: 20.

Question 8:

What will be the output for below javascript code?

console.log('5' - '3');
console.log('5' - 3);
console.log(5 - '3');
Click to view output

Output:

2 for all logs.

Explanation:

In JavaScript, the code console.log("5" - "3"); will output 2. This might seem complex at first, because subtraction typically deals with numbers. However, JavaScript has a behavior where it tries to convert operands (the values used in an operation) to a common type before performing the operation.

In this example, JavaScript attempts to convert both "5" and "3" to numbers before performing the subtraction. Similarly, for all other instances it does the same thing.

Question 9:

What will be the output for below javascript code?

function doSomething() {
  return;
  {
    success: true;
  }
}
console.log(doSomething());
Click to view output

Output:

undefined

Explanation:

In JavaScript, the automatic semicolon insertion (ASI) mechanism interprets the line break after return as the end of the statement, effectively treating it as return;. This means that the function returns undefined and the subsequent object literal is never reached.

When the return statement is encountered in the doSomething function, the function immediately exits and returns undefined by default (if no value is explicitly returned). The object literal { success: true } on the next line is never evaluated or returned because the function has already exited.

Question 10:

What will be the output for below javascript code?

console.log(false ?? 'Some Truthy Value');
console.log(undefined ?? 'Some Truthy Value');
console.log(null ?? 'Some Truthy Value');
console.log(NaN ?? 'Some Truthy Value');
Click to view output

Output:

false Some Truthy Value Some Truthy Value NaN

Explanation:

This is a concept of javascript called, Nullish coalescing operator ??. So as per MDN definition, The nullish coalescing (??) operator is a logical operator that returns its right-hand side operand when its left-hand side operand is null or undefined, and otherwise returns its left-hand side operand.

So whenever the left operand is either undefined or null it will return the right operand as an output or else the left operand itself will be returned.

Question 11:

What will be the output for below javascript code?

console.log(false || 'Some Truthy Value');
console.log(undefined || 'Some Truthy Value');
console.log(null || 'Some Truthy Value');
console.log(NaN || 'Some Truthy Value');
console.log(NaN || NaN || false);
console.log(true || 'Some Truthy Value' || false);
Click to view output

Output:

Some Truthy Value Some Truthy Value Some Truthy Value Some Truthy Value false true

Explanation:

The logical OR operator (||) evaluates the operands from left to right and returns the first truthy value it encounters. If no truthy value is found, it returns the last operand. This behaviour is often used for providing default values.

Question 12:

What will be the output for below javascript code?

console.log(0.1 + 0.2);
console.log(0.1 + 0.2 == 0.3);
Click to view output

Output:

0.30000000000000004 false

Explanation:

console.log(0.1 + 0.2); This line performs an addition operation between two decimal numbers, 0.1 and 0.2. It then logs the result of this operation to the console.

In most programming languages, including JavaScript, floating-point arithmetic operations can sometimes produce unexpected or imprecise results due to the way computers represent and store decimal numbers internally. This is because decimal numbers like 0.1 and 0.2 cannot be represented precisely in binary format.

console.log(0.1 + 0.2 == 0.3) This line also performs the addition 0.1 + 0.2, but it then compares the result with the value 0.3 using the strict equality operator ==.

Due to the imprecise representation of floating-point numbers mentioned earlier, the result of 0.1 + 0.2 is not exactly equal to 0.3. Therefore, when you run this line, the output in the console will be false.

Question 13:

What will be the output for below javascript code?

let x = [0, 1, [2]];
x[2][0] = 3;
console.log(x);
Click to view output

Output:

[0, 1, [3]]

Explanation:

  • The variable x was initialized with the array [0, 1, [2]].

  • The expression x[2][0] accessed the nested array [2] inside x, which is located at index 2. Then, it accessed the element at index 0 of that nested array, which was 2.

  • The assignment x[2][0] = 3 replaced the value 2 with 3 inside the nested array.

  • After the modification, the nested array became [3], and the overall x array became [0, 1, [3]].

  • Finally, console.log(x) printed the updated value of x, which is [0, 1, [3]].

Question 14:

What will be the output for below javascript code?

(function () {
  var a = (b = 3);
})();

console.log('a defined? ' + (typeof a !== undefined));
console.log('b defined? ' + (typeof b !== undefined));
console.log('b value is ', b);
console.log('a value is ', a);
Click to view output

Output:

a defined? true

b defined? true

b value is 3

ReferenceError: a is not defined

Explanation:

Since both the a and b is defined inside the functional scope most of the developer would expect both a and b to be undefined outside of the function scope.

Most of the developers think, that var a = b = 3 is equivalent to var b = 3; var a = b;. But thats not how javascript treats this statement. Instead, javascript treats this code as b = 3; var a = b;

So, b is defined without var keyword so it will be accessible outside of the functional scope as it will be treated as a global variable but a is defined with var keyword it can not be accessed outside of the functional scope. and hence it will be Reference Error.

Question 15:

What will be the output for below javascript code?

var myObject = {
  foo: 'bar',
  func: function () {
    var self = this;
    console.log('outer func:  this.foo = ' + this.foo);
    console.log('outer func:  self.foo = ' + self.foo);
    (function () {
      console.log('inner func:  this.foo = ' + this.foo);
      console.log('inner func:  self.foo = ' + self.foo);
    })();
  }
};
myObject.func();
Click to view output

Output:

outer func: this.foo = bar

outer func: self.foo = bar

inner func: this.foo = undefined

inner func: self.foo = bar

Explanation:

In the outer function, both this and self refer to myObject and therefore both can properly reference and access foo.

In the inner function, though, this no longer refers to myObject. As a result, this.foo is undefined in the inner function, whereas the reference to the local variable self remains in scope and is accessible there.

Question 16:

What will be the output for below javascript code?

function foo1() {
  return {
    bar: 'hello'
  };
}

function foo2() {
  return;
  {
    bar: 'hello';
  }
}

console.log('foo1 returns:');
console.log(foo1());
console.log('foo2 returns:');
console.log(foo2());
Click to view output

Output:

foo1 returns:

Object {bar: "hello"}

foo2 returns:

undefined

Explanation:

In JavaScript, the automatic semicolon insertion (ASI) mechanism interprets the line break after return as the end of the statement, effectively treating it as return;. This means that the function returns undefined and the subsequent object literal is never reached.

When the return statement is encountered in the foo1() function, the function immediately returns the { bar: "hello" }. but in 'foo2()' return statement will be converted to return;, hence will return undefined.

Question 17:

What will be the output for below javascript code?

console.log(Boolean([]));
console.log(Boolean({}));
console.log(Boolean(''));
console.log(Boolean(0));
Click to view output

Output:

true

true

false

false

Explanation:

1 - An empty array [] is considered a truthy value in JavaScript. When the Boolean function is called with an array, it returns true because arrays are considered non-empty objects.

2 - An empty object {} is also considered a truthy value in JavaScript. When the Boolean function is called with an object, it returns true because objects are non-empty.

3 - An empty string "" is considered a falsy value in JavaScript. When the Boolean function is called with an empty string, it returns false.

4 - The number 0 is considered a falsy value in JavaScript. When the Boolean function is called with 0, it returns false.

Question 18:

What will be the output for below javascript code?

function left() {
  return console.log('left');
}

function right() {
  return console.log('right');
}

left() || right();
Click to view output

Output:

left right

Explanation:

In this code, you have two functions, left() and right(), that simply log the strings left and right to the console, respectively. Then, you have the expression left() || right();.

The logical OR operator (||) in JavaScript works in the following way:

1 - It evaluates the operand on the left side, in this case, left().

2 - If the left operand is truthy, it returns the value of the left operand.

3 - If the left operand is falsy, it evaluates the right operand, in this case, right().

4 - If the right operand is evaluated, it returns the value of the right operand.

In this specific case, when left() || right(); is executed, the following happens:

1 - left() is called, which logs left to the console.

2 - The console.log("left"); statement itself returns undefined, which is a falsy value.

3 - Since the left operand left() is falsy, the logical OR operator evaluates the right operand right().

4 - right() is called, which logs right to the console.

5 - The console.log("right"); statement also returns undefined, which is a falsy value.

6 - Since both operands are falsy, the expression left() || right(); evaluates to the value of the right operand, which is undefined.

Therefore, the output of this code will be:

left right

Question 19:

What will be the output for below javascript code?

function counter() {
  let count = 0;
  return () => count++;
}

let c = counter();
console.log(c());
console.log(c());
console.log(c());
Click to view output

Output:

0 1 2

Explanation:

The code defines a function counter that returns another function. The returned function is a closure that has access to the count variable defined inside the counter function.

1 - The counter function is defined.

2 - Inside counter, a variable count is declared and initialized to 0.

3 - The counter function returns an anonymous arrow function () => count++.

  • This anonymous function has access to the count variable through closure.
  • It increments the count variable and returns its value before incrementing.

4 - The counter function is called, and its returned anonymous function is assigned to the variable c.

5 - console.log(c()); is executed three times:

  • The first time, it logs 0 and increments count to 1.
  • The second time, it logs 1 and increments count to 2.
  • The third time, it logs 2 and increments count to 3.

This behavior is possible because the anonymous function returned by counter has access to the count variable through closure. Each time the anonymous function is called, it increments and returns the count variable, which is preserved in the closure.

Closures allow functions to have private variables and maintain state across multiple function calls. In this example, the counter function creates a new private count variable each time it is called, and the returned anonymous function has access to that private variable through the closure.

Question 20:

What will be the output for below javascript code?

const greeting = 'hello';
greeting.length = 10;
console.log(greeting.length);
console.log(greeting);
Click to view output

Output:

5 hello

Explanation:

In JavaScript, strings are immutable, which means their values cannot be changed once they are created. The program demonstrates this behavior by attempting to modify the length property of a string.

1 - The program declares a constant variable greeting and assigns it the string value "hello".

2 - It then attempts to change the length property of the greeting string by assigning it the value 10 using greeting.length = 10;.

3 - However, this assignment operation has no effect because strings in JavaScript are immutable. The length property of a string is a read-only property that reflects the actual number of characters in the string.

4 - The program then logs the length property of the greeting string using console.log(greeting.length);. Since the previous attempt to modify the length property failed, the output is still 5, which is the original length of the "hello" string.

5 - Finally, the program logs the actual value of the greeting string using console.log(greeting);. As expected, the output is "hello", which is the original string value assigned to greeting.

Key Points:

  • Strings in JavaScript are immutable, meaning their values cannot be changed after they are created.
  • The length property of a string is read-only and reflects the actual number of characters in the string.
  • Attempting to modify the length property of a string has no effect, as it is a derived property based on the string's content.

Question 21:

What will be the output for below javascript code?

(function () {
  console.log(1);
  setTimeout(function () {
    console.log(2);
  }, 1000);
  setTimeout(function () {
    console.log(3);
  }, 0);
  console.log(4);
})();
Click to view output

Output:

1 4 3 2

Explanation:

  • 1 and 4 are displayed first since they are logged by simple calls to console.log() without any delay

  • 2 is displayed after 3 because 2 is being logged after a delay of 1000 msecs (i.e., 1 second) whereas 3 is being logged after a delay of 0 msecs.

OK, fine. But if 3 is being logged after a delay of 0 msecs, doesn’t that mean that it is being logged right away? And, if so, shouldn’t it be logged before 4, since 4 is being logged by a later line of code?

The answer has to do with properly understanding JavaScript events and timing.

The browser has an event loop which checks the event queue and processes pending events. For example, if an event happens in the background (e.g., a script onload event) while the browser is busy (e.g., processing an onclick), the event gets appended to the queue. When the onclick handler is complete, the queue is checked and the event is then handled (e.g., the onload script is executed).

Similarly, setTimeout() also puts execution of its referenced function into the event queue if the browser is busy.

When a value of zero is passed as the second argument to setTimeout(), it attempts to execute the specified function “as soon as possible”. Specifically, execution of the function is placed on the event queue to occur on the next timer tick. Note, though, that this is not immediate; the function is not executed until the next tick. That’s why in the above example, the call to console.log(4) occurs before the call to console.log(3) (since the call to console.log(3) is invoked via setTimeout, so it is slightly delayed).

Question 22:

What will be the output for below javascript code?

var arr1 = 'john'.split('');
var arr2 = arr1.reverse();
var arr3 = 'jones'.split('');
arr2.push(arr3);
console.log('array 1: length=' + arr1.length + ' last=' + arr1.slice(-1));
console.log('array 2: length=' + arr2.length + ' last=' + arr2.slice(-1));
Click to view output

Output:

array 1: length=5 last=j,o,n,e,s

array 2: length=5 last=j,o,n,e,s

Explanation:

arr1 and arr2 are the same (i.e. ['n','h','o','j', ['j','o','n','e','s'] ]) after the above code is executed for the following reasons:

  • Calling an array object’s reverse() method doesn’t only return the array in reverse order, it also reverses the order of the array itself (i.e., in this case, arr1).

  • The reverse() method returns a reference to the array itself (i.e., in this case, arr1). As a result, arr2 is simply a reference to (rather than a copy of) arr1. Therefore, when anything is done to arr2 (i.e., when we invoke arr2.push(arr3);), arr1 will be affected as well since arr1 and arr2 are simply references to the same object.

And a couple of side points here that can sometimes trip someone up in answering this question:

  • Passing an array to the push() method of another array pushes that entire array as a single element onto the end of the array. As a result, the statement arr2.push(arr3); adds arr3 in its entirety as a single element to the end of arr2 (i.e., it does not concatenate the two arrays, that’s what the concat() method is for).

  • Like Python, JavaScript honors negative subscripts in calls to array methods like slice() as a way of referencing elements at the end of the array; e.g., a subscript of -1 indicates the last element in the array, and so on.

Question 23:

What will be the output for below javascript code?

console.log(1 + '2' + '2');
console.log(1 + +'2' + '2');
console.log(1 + -'1' + '2');
console.log(+'1' + '1' + '2');
console.log('A' - 'B' + '2');
console.log('A' - 'B' + 2);
Click to view output

Output:

122 32 02 112 NaN2 NaN

Explanation:

Example 1:

1 + "2" + "2" Outputs: "122" Explanation: The first operation to be performed in 1 + "2". Since one of the operands ("2") is a string, JavaScript assumes it needs to perform string concatenation and therefore converts the type of 1 to "1", 1 + "2" yields "12". Then, "12" + "2" yields "122".

Example 2:

1 + +"2" + "2" Outputs: "32" Explanation: Based on order of operations, the first operation to be performed is +"2" (the extra + before the first "2" is treated as a unary operator). Thus, JavaScript converts the type of "2" to numeric and then applies the unary + sign to it (i.e., treats it as a positive number). As a result, the next operation is now 1 + 2 which of course yields 3. But then, we have an operation between a number and a string (i.e., 3 and "2"), so once again JavaScript converts the type of the numeric value to a string and performs string concatenation, yielding "32".

Example 3:

1 + -"1" + "2" Outputs: "02" Explanation: The explanation here is identical to the prior example, except the unary operator is - rather than +. So "1" becomes 1, which then becomes -1 when the - is applied, which is then added to 1 yielding 0, which is then converted to a string and concatenated with the final "2" operand, yielding "02".

Example 4:

+"1" + "1" + "2" Outputs: "112" Explanation: Although the first "1" operand is typecast to a numeric value based on the unary + operator that precedes it, it is then immediately converted back to a string when it is concatenated with the second "1" operand, which is then concatenated with the final "2" operand, yielding the string "112".

Example 5:

"A" - "B" + "2" Outputs: "NaN2" Explanation: Since the - operator can not be applied to strings, and since neither "A" nor "B" can be converted to numeric values, "A" - "B" yields NaN which is then concatenated with the string "2" to yield “NaN2”.

Example 6:

"A" - "B" + 2 Outputs: NaN Explanation: As exlained in the previous example, "A" - "B" yields NaN. But any operator applied to NaN with any other numeric operand will still yield NaN.

Question 24:

What will be the output for below javascript code?

function getValue1() {
  this.value = 10;
  function printValue() {
    console.log(this.value);
  }

  printValue();
}

function getValue2() {
  this.value = 10;
  var printValue = () => {
    console.log(this.value);
  };

  printValue();
}

var value = 20;
new getValue1();
new getValue2();
Click to view output

Output:

undefined 10

Explanation:

The provided JavaScript code demonstrates the concept of this keyword and its behavior in different contexts, specifically within regular functions and arrow functions.

  • In the getValue1 function, this.value is assigned the value of 10. However, inside the nested printValue function, the this keyword refers to the global object (window in browsers) because regular functions create their own this binding. When printValue is called, it logs the value of this.value from the global context, which is initially undefined (because no such property exists on the global object). Therefore, the output will be undefined.

  • In the getValue2 function, this.value is assigned the value of 10, similar to getValue1. However, the printValue function is defined as an arrow function. Arrow functions do not create their own this binding. Instead, they inherit the this value from the surrounding lexical scope, which in this case is the getValue2 function. When printValue is called, it logs the value of this.value from the getValue2 function's context, which is 10.

Question 25:

What will be the output for below javascript code?

console.log('0 || 1 = ' + (0 || 1));
console.log('1 || 2 = ' + (1 || 2));
console.log('0 && 1 = ' + (0 && 1));
console.log('1 && 2 = ' + (1 && 2));
Click to view output

Output:

0 || 1 = 1

1 || 2 = 1

0 && 1 = 0

1 && 2 = 2

Explanation:

In JavaScript, both || and && are logical operators that return the first fully-determined logical value when evaluated from left to right.

The or || operator. In an expression of the form X||Y, X is first evaluated and interpreted as a boolean value. If this boolean value is true, then true 1 is returned and Y is not evaluated, since the or condition has already been satisfied. If this boolean value is false, though, we still don’t know if X||Y is true or false until we evaluate Y, and interpret it as a boolean value as well.

Accordingly, 0 || 1 evaluates to true 1, as does 1 || 2.

The and && operator. In an expression of the form X&&Y, X is first evaluated and interpreted as a boolean value. If this boolean value is false, then false 0 is returned and Y is not evaluated, since the and condition has already failed. If this boolean value is true, though, we still don’t know if X&&Y is true or false until we evaluate Y, and interpret it as a boolean value as well.

However, the interesting thing with the && operator is that when an expression is evaluated as true, then the expression itself is returned. This is fine, since it counts as true in logical expressions, but also can be used to return that value when you care to do so. This explains why, somewhat surprisingly, 1 && 2 returns 2 (whereas you might it expect it to return true or 1).

Question 26:

What will be the output for below javascript code?

console.log(false == '0');
console.log(false === '0');
Click to view output

Output:

true false

Explanation:

In JavaScript, there are two sets of equality operators. The triple-equal operator === behaves like any traditional equality operator would: evaluates to true if the two expressions on either of its sides have the same type and the same value. The double-equal operator, however, tries to coerce the values before comparing them. It is therefore generally good practice to use the === rather than ==. The same holds true for !== vs !=.

Question 27:

What will be the output for below javascript code?

var a = {},
  b = { key: 'b' },
  c = { key: 'c' };

a[b] = 123;
a[c] = 456;

console.log(a[b]);
Click to view output

Output:

456

Explanation:

When setting an object property, JavaScript will implicitly stringify the parameter value. In this case, since b and c are both objects, they will both be converted to [object Object]. As a result, a[b] and a[c] are both equivalent to a["[object Object]"] and can be used interchangeably. Therefore, setting or referencing a[c] is precisely the same as setting or referencing a[b].

Question 28:

What will be the output for below javascript code?

var a = 2;
a++;
console.log(a);

const d = [1, 2, 3];
d.push(5);
console.log(d);

const b = 2;
b++;
console.log(b);

const c = [2];
c[0]++;
console.log(c);
Click to view output

Output:

3

[1, 2, 3, 5]

TypeError: Assignment to constant variable.

Explanation:

  • The reason why the output is 3 is because the increment operation a++ is performed before the console.log(a) statement. If the increment operator was used before the variable (++a), it would return the incremented value (3).

  • Even though d is a constant variable, the contents of the array can be modified. The constant nature of d only prevents re-assigning a new value to the variable itself, but it doesn't make the array immutable. Therefore, when you run this code, it will output: [1, 2, 3, 5]

  • Since b is a constant variable, b++ is trying to assign the incremented value to a constant. which is not allowed, since it will throw an error. TypeError: Assignment to constant variable.

  • And Since, we got an error in our program, 4 set of problem wont be executed. But if you will run that individually then it is going to output [3]

Question 29:

What will be the output for below javascript code?

var a = b();
console.log(a);

var c = d();
function b() {
  return c;
}
console.log(a);

var d = function () {
  return b();
};
console.log(d);
Click to view output

Output:

3

[1, 2, 3, 5]

TypeError: Assignment to constant variable.

Explanation:

  • The reason why the output is 3 is because the increment operation a++ is performed before the console.log(a) statement. If the increment operator was used before the variable (++a), it would return the incremented value (3).

  • Even though d is a constant variable, the contents of the array can be modified. The constant nature of d only prevents re-assigning a new value to the variable itself, but it doesn't make the array immutable. Therefore, when you run this code, it will output: [1, 2, 3, 5]

  • Since b is a constant variable, b++ is trying to assign the incremented value to a constant. which is not allowed, since it will throw an error. TypeError: Assignment to constant variable.

  • And Since, we got an error in our program, 4 set of problem wont be executed. But if you will run that individually then it is going to output [3]

Question 30:

What will be the output for below javascript code?

const promise = new Promise((resolve, reject) => resolve());
promise.finally(() => console.log('cleanup'));
promise.then(() => console.log('do me first!'));
Click to view output

Output:

cleanup

do me first!

Explanation:

  • A promise is created and immediately resolved.
  • The finally() method is called, registering a callback for cleanup, the finally block of code will be executed once the promise is resolved.
  • Then then() will be executed, and as we know already that the promise is resolved, it will log the message do me first!.

Question 31:

What will be the output for below javascript code?

const promise = new Promise((resolve, reject) => resolve());
promise.finally(() => promise.finally(() => console.log('cleanup')));
promise.then(() => console.log('do me first!'));
Click to view output

Output:

do me first!

cleanup

Explanation:

The order of execution is important here...

  • The Promise is resolved immediately.
  • The first finally callback is called, which calls the second finally callback.
  • The then callback is called after the first finally callback.
  • So, The do me first! is logged first because the then callback is called before the nested finally callback that logs cleanup.

Note: It's worth nothing that the finally callback is typically used for cleanup operations or logging, and it doesn't affect the resolved value or the rejection reason of the Promise chain.

Question 32:

What will be the output for below javascript code?

function foo() {
  console.log(1);
}

foo();

foo = function () {
  console.log(2);
};
Click to view output

Output:

1

Explanation:

The code you provided demonstrates a concept in JavaScript called "Function Hoisting." Here's a summary of what's happening:

  • function foo() { console.log(1) } declares a function named foo and assigns it a function expression that logs the number 1 to the console.
  • foo(); calls the foo function, which outputs 1 in the console.
  • foo = function() { console.log(2); } attempts to reassign the foo variable with a new anonymous function expression that logs the number 2 to the console.

However, due to Function Hoisting, the initial function declaration function foo() { console.log(1) } is hoisted to the top of the scope (in this case, the global scope) during the compilation phase. This means that when foo(); is called, it executes the original foo function that logs 1.

After that, the reassignment foo = function() { console.log(2); } overwrites the original foo function with the new anonymous function expression. If you were to call foo() again after this reassignment, it would log 2 instead of 1.

Above code demonstrates that function declarations are hoisted to the top of their scope during the compilation phase, while function expressions are not hoisted until they are executed during the runtime phase.

Question 33:

What will be the output for below javascript code?

function foo() {
  var a = 2;
  function bar() {
    console.log(a);
  }
  bar();
}
foo();
Click to view output

Output:

2

Explanation:

The provided code demonstrates the concept of lexical scoping in JavaScript. Here's a breakdown of what's happening:

  • function foo() { ... } declares a function named foo.
  • Inside foo, the variable a is declared and assigned the value 2 using var a = 2;.
  • Another function named bar is declared inside foo using function bar() { console.log(a) }. The bar function logs the value of a to the console.
  • bar(); is called, which executes the console.log(a) statement inside bar.
  • foo(); is called, which triggers the execution of the foo function.

When bar() is called inside foo, it can access the variable a because of lexical scoping. Lexical scoping means that a function has access to variables defined in its outer (parent) function scope, even after the outer function has finished executing.

In this case, bar is defined inside the scope of foo, so it has access to the a variable declared in foo. Therefore, when bar() is called, it logs the value of a, which is 2.

Question 34:

What will be the output for below javascript code?

setTimeout(function () {
  console.log('d');
}, 0);

console.log('c');

setTimeout(function () {
  console.log('A');
}, 2000);

setTimeout(function () {
  console.log('b');
}, 1000);
Click to view output

Output:

c d b A

Explanation:

  • setTimeout(function() { console.log('d'); }, 0); schedules a function that logs d to be executed after a delay of 0 milliseconds (essentially, as soon as possible, but not immediately).
  • console.log('c'); logs c to the console immediately.
  • setTimeout(function() { console.log('A'); }, 2000); schedules a function that logs A to be executed after a delay of 2000 milliseconds (2 seconds).
  • setTimeout(function() { console.log('b'); }, 1000); schedules a function that logs b to be executed after a delay of 1000 milliseconds (1 second).

The order of execution is as follows:

  • c is logged immediately because it's not inside a setTimeout function.
  • d is logged next, even though it was scheduled with a delay of 0 milliseconds. This is because the setTimeout callbacks are pushed into the Event Queue and are executed after the current execution stack is empty.
  • b is logged after a delay of 1 second, as scheduled.
  • A is logged after a delay of 2 seconds, as scheduled.

It's important to note that setTimeout is an asynchronous operation, which means that the scheduled functions are not executed on the main execution thread. Instead, they are added to the Event Queue and executed when the main execution stack is empty.

This behavior ensures that the execution of long-running tasks does not block the main thread, allowing the browser to remain responsive to user interactions and other events.

Question 35:

What will be the output for below javascript code?

var output = (function (x) {
  delete x;
  return x;
})(0);

console.log(output);
Click to view output

Output:

0

Explanation:

  • The code creates an immediately invoked function expression (IIFE) that takes a single parameter x. The IIFE is executed immediately, and the value 0 is passed as an argument.
  • Inside the IIFE, delete x; is executed. The delete operator is used to remove a property from an object or delete an element from an array. However, when used with a variable that holds a primitive value (like a number, string, or boolean), the delete operator has no effect. It doesn't delete the variable or its value.
  • The IIFE then returns the value of x, which is still 0 because the delete operation had no effect.
  • The returned value 0 is assigned to the variable output.
  • Finally, console.log(output); logs the value of output, which is 0.

the delete operator cannot remove variables that hold primitive values in JavaScript. It can only remove object properties or array elements. When used with a primitive value, it simply has no effect, and the value remains unchanged.

Therefore, the output of this code will be 0.

Question 36:

What will be the output for below javascript code?

var a;
console.log(a);

function test() {
  a = 1;
}
test();

console.log(a);
Click to view output

Output:

undefined 1

Explanation:

  • The variable a is declared using var a;. Due to variable hoisting, this declaration is moved to the top of the scope (in this case, the global scope) during the compilation phase. However, the assignment is not hoisted, so a is initially undefined.
  • console.log(a); logs the current value of a, which is undefined.
  • The test function is declared. Function declarations are hoisted to the top of the scope in their entirety during the compilation phase.
  • test(); calls the test function, which assigns the value 1 to the variable a.
  • console.log(a); logs the updated value of a, which is now 1.

Question 37:

What will be the output for below javascript code?

foo();
bar();

var foo = function () {
  console.log('foo');
};

function bar() {
  console.log('bar');
}
Click to view output

Output:

TypeError: foo is not a function

Explanation:

  • foo(); is called before the foo function is declared or assigned. Due to variable hoisting, the variable foo is declared at the top of the scope and initialised with undefined. However, when the code tries to invoke foo() as a function, it throws a TypeError: foo is not a function because foo is undefined at this point, and undefined is not a function.
  • Since the error occurred in the program, the program will be terminated.

Question 38:

What will be the output for below javascript code?

function a() {
  for (var i = 0; i < 10; i++) {
    setTimeout(() => {
      console.log(i);
    }, i * 1000);
  }
}
a();
Click to view output

Output:

10 (10 times on every second)

Explanation:

The provided code demonstrates the behavior of closures and asynchronous execution in JavaScript. When you run this code, it will output the numbers from 0 to 9 with a delay of 1 second between each output. However, all the numbers will be printed one after another, instead of being printed with the expected delay.

  • The a function is called, which initiates the for loop.
  • Inside the loop, for each iteration, a new timeout is set using setTimeout. The timeout function is an arrow function that logs the value of i to the console.
  • The setTimeout function is asynchronous, which means it doesn't block the execution of the code. Instead, it schedules the callback function (the arrow function () => { console.log(i); }) to be executed after the specified delay (i * 1000 milliseconds).
  • The for loop continues to execute until it completes, incrementing the value of i for each iteration.
  • After the loop finishes, the setTimeout callbacks are executed one by one, but by this time, the value of i is already 10 (because the loop has completed).

The reason why all the numbers are printed one after another is that the closures created by the setTimeout callbacks capture the same variable i, which is incremented during the loop. By the time the callbacks are executed, the value of i is 10 for all the callbacks, so they all log 10 to the console.

To fix this issue and get the desired behavior (printing 0 after 0 seconds, 1 after 1 second, 2 after 2 seconds, and so on), you need to create a new scope for each iteration of the loop. One way to achieve this is by using an immediately invoked function expression (IIFE) or by using let instead of var (since let has block-level scope):

function a() {
  for (let i = 0; i < 10; i++) {
    setTimeout(() => {
      console.log(i);
    }, i * 1000);
  }
}

a();

Question 39:

What will be the output for below javascript code?

let b = {
  x: 10,
  y: 20
};
let a = JSON.parse(JSON.stringify(b));
a.x = 20;
console.log(a);
console.log(b);
Click to view output

Output:

{x: 20, y: 20} {x: 20, y: 20}

Explanation:

  • let b = { x: 10, y: 20 } creates an object b with two properties: x and y.

  • let a = JSON.parse(JSON.stringify(b)); creates a deep copy of b and assigns it to a. Here's how it works:

    • JSON.stringify(b) converts the object b into a JSON string representation: '{"x":10,"y":20}'.
    • JSON.parse() takes this JSON string and converts it back into a JavaScript object, creating a new object with the same properties and values as b.
  • This process creates a new object in memory, separate from the original b object.

  • a.x = 20; modifies the x property of the a object, but it doesn't affect the b object because a is a separate copy.

  • console.log(a); outputs { x: 20, y: 20 }, reflecting the change made to a.x.

  • console.log(b); outputs { x: 10, y: 20 }, showing that b remains unchanged.

The JSON.parse(JSON.stringify(...)) technique is a common way to create a deep copy of an object in JavaScript. However, it has some limitations:

  • It doesn't work for objects that contain circular references or methods (functions).
  • It doesn't preserve the prototype chain or object class (the copied object will be a plain object {}).
  • It doesn't preserve data types like Date, RegExp, or undefined values (they are converted to their string representations).

Question 40:

What will be the output for below javascript code?

function main() {
  var a = 1000;
  let b = 100;
  if (true) {
    var a = 'tanish';
    let b = 'javascript';
    console.log(a);
    console.log(b);
  }
  console.log(a);
  console.log(b);
}
main();
Click to view output

Output:

tanish javascript tanish 100

Explanation:

The provided code demonstrates the behavior of variable hoisting and block-level scoping in JavaScript.

1 - The function main is defined and declared.

2 - Inside main, the variable a is declared with var and initialized to 1000. Due to variable hoisting, this declaration is moved to the top of the function scope during the compilation phase.

3 - The variable b is declared with let and initialized to 100. let variables have block-level scope and are not hoisted.

4 - Inside the if (true) block:

  • A new variable a is declared with var. However, because of variable hoisting, this doesn't create a new variable; instead, it reassigns the existing a variable to the string tanish.
  • A new variable b is declared with let and initialized to javascript. This b is a different variable from the outer b due to block-level scoping.
  • console.log(a) outputs tanish because a was reassigned inside the block.
  • console.log(b) outputs javascript because it accesses the inner b variable.

5 - Outside the if block:

  • console.log(a) outputs tanish because the reassignment of a inside the block persisted due to variable hoisting.
  • console.log(b) outputs 100 because it accesses the outer b variable, as the inner b variable has block-level scope and is not accessible outside the if block.

This code demonstrates that:

  • Variables declared with var are function-scoped and can be reassigned within the same scope, even if they are declared multiple times.
  • Variables declared with let have block-level scope and are not hoisted. A new let variable inside a block creates a new binding, separate from outer variables with the same name.

Question 41:

What will be the output for below javascript code?

f1();
f2();
function f1() {
  console.log('f1 executed');
}

var f2 = function () {
  console.log('f2 executed');
};
Click to view output

Output:

f1 executed

Uncaught TypeError: f2 is not a function

Explanation:

  • f1(); is called before its declaration. Due to function hoisting, the entire function f1 is hoisted to the top of the scope (in this case, the global scope) during the compilation phase. Therefore, when f1(); is called, it correctly logs f1 executed.
  • f2(); is called before its declaration. However, only the variable declaration var f2; is hoisted to the top of the scope due to variable hoisting. The assignment f2 = function() { ... } is not hoisted. As a result, when f2(); is called, f2 is still undefined, and attempting to invoke an undefined value as a function throws a TypeError: f2 is not a function.
  • After the hoisting, the function declaration function f1() { ... } is executed, defining the f1 function in memory.
  • Finally, the variable assignment var f2 = function() { ... } is executed, assigning the function expression to the variable f2.

Question 42:

What will be the output for below javascript code?

var a = [2, 3, 4, 7];
for (var i = 0; i < a.length; i++) {
  console.log(a[i]);
  setTimeout(() => {
    console.log(a[i]);
  }, 5000);
}
Click to view output

Output:

2 3 4 7

and after 5 seconds

undefined undefined undefined undefined

Explanation:

  • var a = [2, 3, 4, 7]; will create a variable and initialise an array of values.
  • for (var i = 0; i < a.length; i++) will iterate over all the values of array a.
  • console.log(a[i]); will print the value from the array for the given iteration index.
  • setTimeout() method will be executed after 5 seconds and then whatever the value of a[i] would be there will be print on the console.

In this case, Line #3, will log the array values as 2 3 4 7 but as soon as value of i reaches to 4, for loop wont be executed because the condition fails, and then after 5 seconds the values of a[i] which is a[4] i.e undefined will be logged.

Workaround to fix the issue:

var a = [2, 3, 4, 7];
for (var i = 0; i < a.length; i++) {
  ((i) => {
    console.log(a[i]);
    setTimeout(() => {
      console.log(a[i]);
    }, 5000);
  })(i);
}

Another workaround is:

var a = [2, 3, 4, 7];
for (let i = 0; i < a.length; i++) {
  console.log(a[i]);
  setTimeout(() => {
    console.log(a[i]);
  }, 5000);
}

Question 43:

What will be the output for below javascript code?

function callMe(delay) {
  setTimeout(() => {
    console.log('Called');
  }, delay);
}

const delays = [500, 1000, 1500];

for (let delay of delays) {
  callMe(delay);
}

console.log('Completed');
Click to view output

Output:

2 3 4 7

and after 5 seconds

undefined undefined undefined undefined

Explanation:

  • The console.log('Completed'); statement is executed immediately after the for loop finishes because it is not inside an asynchronous callback function.
  • The setTimeout functions scheduled inside callMe are added to the Event Queue, and they will be executed asynchronously once the main execution thread is available.
  • The delay of 500 milliseconds (0.5 seconds) for the first setTimeout expires first, so Called is logged to the console.
  • After another 500 milliseconds (1 second in total), the second setTimeout with a delay of 1000 milliseconds expires, and Called is logged again.
  • Finally, after another 500 milliseconds (1.5 seconds in total), the third setTimeout with a delay of 1500 milliseconds expires, and Called is logged for the third time.

Question 44:

What will be the output for below javascript code?

let randomValue = { name: 'Lydia' };
randomValue = 23;

if (!typeof randomValue === 'string') {
  console.log("It's not a string!");
} else {
  console.log("Yay it's a string!");
}
Click to view output

Output:

Yay it's a string!

Explanation:

  • let randomValue = { name: "Lydia" } declares a variable randomValue and initializes it with an object { name: "Lydia" }.
  • randomValue = 23 reassigns the value of randomValue to the number 23.
  • !typeof randomValue === "string" is an expression that evaluates to false because:
    • typeof randomValue evaluates to number since randomValue is now 23.
    • The ! operator negates the result, since "number" is a truthy value and ! making truthy value evaluate to false.
    • false === "string" evaluates to false.
  • Since the condition !typeof randomValue === "string" evaluates to false, the code block inside the else statement is executed.
  • console.log("Yay it's a string!") logs the string Yay it's a string! to the console.

Question 45:

What will be the output for below javascript code?

const createMember = ({ email, address = {} }) => {
  const validEmail = /.+\@.+\..+/.test(email);
  if (!validEmail) throw new Error('Valid email pls');

  return {
    email,
    address: address ? address : null
  };
};

const member = createMember({
  email: 'my@email.com'
});
console.log(member);
Click to view output

Output:

{ email: "my@email.com", address: {} }

Explanation:

The default value of address is an empty object {}. When we set the variable member equal to the object returned by the createMember function, we didn't pass a value for the address, which means that the value of the address is the default empty object {}. An empty object is a truthy value, which means that the condition of the address ? address : null conditional returns true. The value of the address is the empty object {}.

Question 46:

What will be the output for below javascript code?

const animals = {};
let dog = { emoji: '🐶' };
let cat = { emoji: '🐈' };

animals[dog] = { ...dog, name: 'Mara' };
animals[cat] = { ...cat, name: 'Sara' };

console.log(animals[dog]);
Click to view output

Output:

{emoji: '🐈', name: 'Sara'}

Explanation:

Object keys are converted to strings.

Since the value of dog is an object, animals[dog] actually means that we’re creating a new property called [object Object] equal to the new object. animals["[object Object]"] is now equal to { emoji: "🐶", name: "Mara"}.

cat is also an object, which means that animals[cat] actually means that we’re overwriting the value of animals["[object Object]"] with the new cat properties.

Logging animals[dog], or actually animals["[object Object]"] since converting the dog object to a string results [object Object], returns the { emoji: "🐈", name: "Sara" }.

Question 47:

What will be the output for below javascript code?

const promise1 = Promise.resolve('First');
const promise2 = Promise.resolve('Second');
const promise3 = Promise.reject('Third');
const promise4 = Promise.resolve('Fourth');

const runPromises = async () => {
  const res1 = await Promise.all([promise1, promise2]);
  const res2 = await Promise.all([promise3, promise4]);
  return [res1, res2];
};

runPromises()
  .then((res) => console.log(res))
  .catch((err) => console.log(err));
Click to view output

Output:

Third

Explanation:

The Promise.all method runs the passed promises in parallel. If one promise fails, the Promise.all method rejects with the value of the rejected promise. In this case, promise3 is rejected with the value Third. We’re catching the rejected value in the chained catch method on the runPromises invocation to catch any errors within the runPromises function. Only Third gets logged, since promise3 is rejected with this value.

Question 48:

What will be the output for below javascript code?

const user = {
  email: 'my@email.com',
  updateEmail: (email) => {
    this.email = email;
  }
};

user.updateEmail('new@email.com');
console.log(user.email);
Click to view output

Output:

my@email.com

Explanation:

The updateEmail function is an arrow function, and is not bound to the user object. This means that the this keyword is not referring to the user object, but refers to the global scope in this case. The value of email within the user object does not get updated. When logging the value of user.email, the original value of my@email.com gets returned.

Question 49:

What will be the output for below javascript code?

const fruit = ['🍌', '🍊', '🍎'];

fruit.slice(0, 1);
fruit.splice(0, 1);
fruit.unshift('🍇');

console.log(fruit);
Click to view output

Output:

['🍇', '🍊', '🍎']

Explanation:

  1. const fruit = ['🍌', '🍊', '🍎']: This line initializes an array fruit with three elements: banana, orange, and apple emojis.

  2. fruit.slice(0, 1): This method creates a new array containing a shallow copy of a portion of the original array fruit. In this case, it creates a new array with the first element (['🍌']). However, since slice doesn't modify the original array, there's no visible output.

  3. fruit.splice(0, 1): This method changes the contents of the original array fruit by removing elements from it. The first argument 0 specifies the starting index, and the second argument 1 specifies the number of elements to remove. In this case, it removes the first element ('🍌') from the array fruit, modifying it to ['🍊', '🍎'].

  4. fruit.unshift('🍇'): This method inserts a new element ('🍇') at the beginning of the array fruit. After this operation, the array becomes ['🍇', '🍊', '🍎'].

  5. console.log(fruit): This line logs the current state of the fruit array to the console, which is ['🍇', '🍊', '🍎'].

So, the final output after executing all the operations on the fruit array is ['🍇', '🍊', '🍎'].

Question 50:

What will be the output for below javascript code?

const user = {
  email: 'e@mail.com',
  password: '12345'
};

const updateUser = ({ email, password }) => {
  if (email) {
    Object.assign(user, {
      email
    });
  }

  if (password) {
    user.password = password;
  }

  return user;
};

const updatedUser = updateUser({
  email: 'new@email.com'
});

console.log(updatedUser === user);
Click to view output

Output:

true

Explanation:

  • The user object is initialized with two properties: email and password.
  • The updateUser function takes an object as an argument and uses object destructuring to extract the email and password properties.
  • Inside the updateUser function:
    • If an email is provided, it updates the user object's email property using Object.assign.
    • If a password is provided, it updates the user object's password property directly.
    • Finally, it returns the user object.
  • The updatedUser variable is assigned the result of calling updateUser with an object that has a new email value.
  • The last line console.log(updatedUser === user) compares the updatedUser object with the user object using the strict equality operator ===.

The output of this code will be: true

This is because the updateUser function modifies the original user object directly, and updatedUser is just a reference to the same object. When objects are compared using the strict equality operator ===, it checks if they refer to the same object in memory. Since updatedUser and user point to the same object, the comparison returns true.

Question 51:

What will be the output for below javascript code?

class Calc {
  constructor() {
    this.count = 0;
  }

  increase() {
    this.count++;
  }
}

const calc = new Calc();
new Calc().increase();

console.log(calc.count);
Click to view output

Output:

0

Explanation:

  1. The Calc class is defined with a constructor method that initializes the count property to 0.
  2. The increase method is defined within the Calc class, which increments the count property by 1.
  3. An instance of the Calc class is created and assigned to the calc variable using const calc = new Calc(). This creates a new object with the count property set to 0.
  4. A new instance of the Calc class is created with new Calc(), and the increase method is called on that instance using .increase(). This increments the count property of the new instance by 1, but it does not affect the count property of the calc instance.
  5. The console.log(calc.count) line logs the value of the count property of the calc instance to the console.

The output of this code will be: 0

This is because when you create a new instance of the Calc class with new Calc().increase(), it creates a separate instance from the calc instance. The increase method is called on this new instance, incrementing its count property, but it does not affect the count property of the calc instance.

Therefore, when console.log(calc.count) is executed, it prints 0 because the count property of the calc instance was never incremented.

Question 52:

What will be the output for below javascript code?

function getFruit(fruits) {
  console.log(fruits?.[1]?.[1]);
}

getFruit([['🍊', '🍌'], ['🍍']]);
getFruit();
getFruit([['🍍'], ['🍊', '🍌']]);
Click to view output

Output:

undefined undefined 🍌

Explanation:

  1. The getFruit function takes an optional fruits parameter, which is expected to be an array of arrays.
  2. Inside the function, the expression fruits?.[1]?.[1] uses optional chaining to safely access the elements of the nested arrays.
  • The first ?.[1] attempts to access the second element (index 1) of the outer array fruits. If fruits is null or undefined, the expression short-circuits and returns undefined.
  • The second ?.[1] attempts to access the second element (index 1) of the array retrieved from the first step. If the previous step returned undefined, the expression short-circuits and returns undefined.
  1. The console.log statement logs the value obtained from the optional chaining expression.
  2. The function is called three times with different arguments:
  • getFruit([['🍊', '🍌'], ['🍍']]): This call logs undefined because the second element of the inner array ['🍍'] does not exist.
  • getFruit(): This call logs undefined because fruits is undefined, causing the optional chaining expression to short-circuit.
  • getFruit([['🍍'], ['🍊', '🍌']]): This call logs '🍌' because the second element of the second inner array ['🍊', '🍌'] is '🍌'.

The output of this code will be: undefined undefined 🍌

The optional chaining operator ?. is a safe way to access nested properties or elements of an object or array. It provides a short-circuit behavior that prevents errors when accessing null or undefined values, making it easier to handle potentially missing or undefined data.

Question 53:

What will be the output for below javascript code?

class Bird {
  constructor() {
    console.log("I'm a bird. 🦢");
  }
}

class Flamingo extends Bird {
  constructor() {
    console.log("I'm pink. 🌸");
    super();
  }
}

const pet = new Flamingo();
Click to view output

Output:

I'm pink. 🌸 I'm a bird. 🦢

Explanation:

  • The Bird class is defined with a constructor that logs the message "I'm a bird. 🦢".
  • The Flamingo class is defined as a subclass that extends the Bird class using the extends keyword.
  • Inside the Flamingo class, a constructor is defined.
  • Inside the Flamingo constructor, the first statement logs the message "I'm pink. 🌸".
  • The super() call is made inside the Flamingo constructor. This invokes the constructor of the parent class (Bird).
  • An instance of the Flamingo class is created using const pet = new Flamingo().

The output of this code will be:

I'm pink. 🌸

I'm a bird. 🦢

Here's what happens when const pet = new Flamingo() is executed:

  1. The new keyword is used to create a new instance of the Flamingo class.
  2. The Flamingo constructor is called.
  3. Inside the Flamingo constructor, the statement console.log("I'm pink. 🌸") is executed, and "I'm pink. 🌸" is logged to the console.
  4. The super() call is made, which invokes the constructor of the parent class (Bird).
  5. Inside the Bird constructor, the statement console.log("I'm a bird. 🦢") is executed, and "I'm a bird. 🦢" is logged to the console.

In JavaScript, when creating a subclass that extends a parent class, the super() method must be called before accessing this or returning from the constructor. This ensures that the parent class's constructor is executed and the instance is properly initialized.

In this example, the Flamingo class inherits from the Bird class, and the super() call ensures that the Bird constructor is executed after the Flamingo constructor's initial log statement.

Question 54:

What will be the output for below javascript code?

const person = {
  name: 'Lydia Hallie',
  hobbies: ['coding']
};

function addHobby(hobby, hobbies = person.hobbies) {
  hobbies.push(hobby);
  return hobbies;
}

addHobby('running', []);
addHobby('dancing');
addHobby('baking', person.hobbies);

console.log(person.hobbies);
Click to view output

Output:

['coding', 'dancing', 'baking']

Explanation:

  1. An object person is defined with two properties: name (a string) and hobbies (an array containing one hobby, 'coding').
  2. A function addHobby is defined that takes two parameters:
  • hobby: the new hobby to be added.
  • hobbies: an array of hobbies with a default value of person.hobbies.
  1. Inside the addHobby function:
  • The hobby is pushed into the hobbies array.
  • The modified hobbies array is returned.
  1. addHobby('running', []): This call creates a new empty array [] and passes it as the hobbies argument. The function adds 'running' to this new array and returns ['running']. However, this doesn't affect person.hobbies.
  2. addHobby('dancing'): This call uses the default value person.hobbies for the hobbies parameter. 'dancing' is added to person.hobbies, so now person.hobbies is ['coding', 'dancing'].
  3. addHobby('baking', person.hobbies): This call explicitly passes person.hobbies as the hobbies argument. 'baking' is added to person.hobbies, so now person.hobbies is ['coding', 'dancing', 'baking'].
  4. console.log(person.hobbies): This logs the current value of person.hobbies to the console.

The output of this code will be: ['coding', 'dancing', 'baking']

The key point here is that in JavaScript, objects (including arrays) are passed by reference. When you pass person.hobbies as an argument or use it as the default value, any modifications to hobbies inside the addHobby function will directly affect person.hobbies.

In this case:

  • The first call to addHobby doesn't affect person.hobbies because a new empty array is provided.
  • The second and third calls modify person.hobbies directly, adding 'dancing' and 'baking' respectively.

This demonstrates how default parameters and object mutations can lead to side effects in functions.

Question 55:

What will be the output for below javascript code?

class Counter {
  #number = 10;

  increment() {
    this.#number++;
  }

  getNum() {
    return this.#number;
  }
}

const counter = new Counter();
counter.increment();

console.log(counter.#number);
Click to view output

Output:

11

Explanation:

In ES2020 (ECMAScript 2020), JavaScript introduced the concept of private class fields using the # symbol. This feature provides true privacy for class properties, a capability that was previously simulated using naming conventions like prefixing properties with an underscore (_).

Question 56:

What will be the output for below javascript code?

const obj = {
  1: 'a',
  1: 'b',
  [1]: 'c'
};

console.log(obj['1']);
Click to view output

Output:

c

Explanation:

In JavaScript, object keys are either strings or symbols. When a non-string key is used, it is converted to a string.

In the given code, the object obj is defined with three keys:

  • "1": "a"
  • 1: "b"
  • [1]: "c"

However, 1 and [1] are numbers, which are automatically converted to the string "1".

This results in the object being defined as:

const obj = {
  1: 'c'
};

Here's how the transformation happens step by step:

  1. "1": "a" - Adds a key "1" with value "a".
  2. 1: "b" - The key 1 is converted to the string "1", thus it overwrites the previous value "a" with "b".
  3. [1]: "c" - The expression [1] evaluates to 1, which again is converted to the string "1", overwriting the previous value "b" with "c".

So, when we access obj["1"], the value is "c".

Therefore, the output is c.

Question 57:

What will be the output for below javascript code?

const languages = ['java', 'javasript'];

const obj = {
  ...languages
};

console.log(obj);

console.log('javascript' in obj);
console.log('1' in languages);
Click to view output

Output:

false true

Explanation:

The in operator in console.log("javascript" in obj); checks if a property exists in an object. Here, it checks if javascript is a key in obj. Since obj has keys 0 and 1, but not javascript, the output will be: false

The in operator in console.log("1" in languages); checks if a property exists in an array. Here, it checks if the string "1" is a valid index in the languages array. Since languages has an element at index 1, the output will be: true

Question 58:

What will be the output for below javascript code?

let a = [1, 2, 3, 4, 5, 6];
a[a.length - 1]++;
console.log(a);
Click to view output

Output:

[1, 2, 3, 4, 5, 7]

Explanation:

Here, a.length - 1 calculates the index of the last element in the array, which is 5 (since array indices are zero-based and the array has 6 elements). So, a[5] refers to the last element, which is 6. The ++ operator increments this element by 1, changing its value from 6 to 7.

Question 59:

What will be the output for below javascript code?

if (undefined) {
  console.log(1);
} else {
  console.log(2);
}
Click to view output

Output:

2

Explanation:

In JavaScript, undefined is a falsy value. The if statement evaluates the condition undefined, which is false. Therefore, the code inside the else block is executed, resulting in the output: 2

Question 60:

What will be the output for below javascript code?

let arr = [1, 2, 3, 4, 5, 6, 7];
arr.length = 0;
console.log(arr);
Click to view output

Output:

[]

Explanation:

Setting arr.length = 0 clears the array. Thus, console.log(arr); outputs: []

Question 61:

What will be the output for below javascript code?

const arr = ['A', 'B', 'C', 'D', 'E'];
console.log(Object.keys(arr));
Click to view output

Output:

["0", "1", "2", "3", "4"]

Explanation:

Object.keys(arr) returns an array of the keys (indices) of the arr array. which will output: ["0", "1", "2", "3", "4"]

Question 62:

What will be the output for below javascript code?

var a = 10;
let a = 20;
console.log(a);
Click to view output

Output:

SyntaxError: Identifier 'a' has already been declared

Explanation:

This code will cause a syntax error: SyntaxError: Identifier 'a' has already been declared.

Question 63:

What will be the output for below javascript code?

const person = {
  firstName: 'Tanish'
};
const { firstName = 'Hinu' } = person;
console.log(firstName);
Click to view output

Output:

Tanish

Explanation:

  • The person object has a property firstName with the value 'Tanish'.
  • The destructuring assignment const { firstName = "Hinu" } = person; attempts to extract firstName from person. If firstName is undefined, it would use the default value "Hinu".
  • However, since firstName is already defined in the person object with the value 'Tanish', the default value "Hinu" is not used.

Question 64:

What will be the output for below javascript code?

const arr1 = [1, 2, 3, 4];
const arr2 = [6, 7, 5];
const result = [...arr1, ...arr2];
console.log(result);
Click to view output

Output:

[1, 2, 3, 4, 6, 7, 5]

Explanation:

  • The spread operator (...) is used to unpack the elements of arr1 and arr2 into a new array result.
  • The result array contains all the elements from arr1 followed by all the elements from arr2. which is [1, 2, 3, 4, 6, 7, 5]

Question 65:

What will be the output for below javascript code?

let x = 5;
let y = x++;
console.log(y);
console.log(x);
Click to view output

Output:

5 6

Explanation:

  • let x = 5; initializes x with the value 5.
  • let y = x++; assigns the current value of x to y and then increments x by 1.
    • y gets the value 5.
    • x is incremented to 6.

Question 66:

What will be the output for below javascript code?

let x = 5;
let y = ++x;
console.log(y);
console.log(x);
Click to view output

Output:

6 6

Explanation:

  • let x = 5; initializes x with the value 5.
  • let y = ++x; increments x by 1 first and then assigns the new value of x to y.
    • x is incremented to 6.
    • y gets the value 6.

Question 67:

What will be the output for below javascript code?

const a = { x: 1 };
const b = { x: 1 };
console.log(a === b);
console.log(a.x === b.x);
Click to view output

Output:

false true

Explanation:

  • const a = { x: 1 }; initializes a with an object containing a property x with the value 1.
  • const b = { x: 1 }; initializes b with another object containing a property x with the value 1.
  • console.log(a === b);
    • This checks for strict equality between the objects a and b.
    • In JavaScript, objects are compared by reference, not by value. Since a and b are different objects (even though they have the same properties and values), the result is false.
  • console.log(a.x === b.x);
    • This checks for strict equality between the values of the x properties of a and b.
    • Since both a.x and b.x have the value 1, the result is true.

Question 68:

What will be the output for below javascript code?

const arr = [11, 0, '', false, 2, 1];
const filtered = arr.filter(Boolean);
console.log(filtered);
Click to view output

Output:

[11, 2, 1]

Explanation:

  • const arr = [11, 0, '', false, 2, 1]; initializes an array arr with the elements [11, 0, '', false, 2, 1].
  • const filtered = arr.filter(Boolean); uses the filter method with Boolean as the callback function. The Boolean function returns false for falsy values (0, '', false, null, undefined, NaN) and true for truthy values.
    • The filter method creates a new array with all the elements that pass the test implemented by the provided function.
    • In this case, it filters out the falsy values from arr.

Question 69:

What will be the output for below javascript code?

var a = 10;
let a = 20;
console.log(a);
Click to view output

Output:

SyntaxError: Identifier 'a' has already been declared

Explanation:

This code snippet demonstrates a common JavaScript error:

  1. var a = 10; declares a variable a with function scope.
  2. let a = 20; attempts to redeclare a with block scope.
  3. This causes a SyntaxError because let doesn't allow redeclaration of variables in the same scope.
  4. console.log(a); is never reached due to the error.

Question 70:

What will be the output for below javascript code?

const arr = ['A', 'B', 'C', 'D', 'E'];
console.log(Object.keys(arr));
Click to view output

Output:

['0', '1', '2', '3', '4']

Explanation:

This code snippet demonstrates a common JavaScript error:

  1. An array arr is created with five string elements.
  2. Object.keys(arr) returns an array of the array's indices as strings.
  3. The console will log: ['0', '1', '2', '3', '4']

Question 71:

What will be the output for below javascript code?

var trees = ['pine', 'apple', 'oak', 'maple', 'cherry'];
delete trees[3];
console.log(trees.length);
console.log(trees);
Click to view output

Output:

5 ['pine', 'apple', 'oak', empty, 'cherry']

Explanation:

  1. var trees = ["pine","apple","oak","maple","cherry"]; creates an array with 5 elements.
  2. delete trees[3]; removes the value at index 3 ("maple"), leaving an undefined slot.
  3. console.log(trees.length); outputs 5.
  4. The delete operator removes the element but doesn't affect the array's length.

Question 72:

What will be the output for below javascript code?

var a = [1, 2, 3];
a[10] = 99;
console.log(a[6]);
Click to view output

Output:

undefined

Explanation:

  1. var a = [1, 2, 3]; initializes an array with 3 elements.
  2. a[10] = 99; assigns a value to index 10, expanding the array.
  3. console.log(a[6]); outputs undefined.
  4. Accessing an index between 3 and 9 returns undefined as these are empty slots.

Question 73:

What will be the output for below javascript code?

var Employee = {
  company: 'Acme'
};
var employee1 = Object.create(Employee);
delete employee1.company;
console.log(employee1.company);
Click to view output

Output:

Acme

Explanation:

  1. var Employee = { company: 'Acme' } creates an object with a company property.
  2. var employee1 = Object.create(Employee); creates a new object with Employee as its prototype.
  3. delete employee1.company attempts to delete the company property from employee1.
  4. console.log(employee1.company); outputs 'Acme'.
  5. The delete operation doesn't affect the prototype chain, so company is still accessible from the prototype.

Question 74:

What will be the output for below javascript code?

var z = 1,
  y = (z = typeof y);
console.log(y);
Click to view output

Output:

undefined

Explanation:

  1. var z = 1, y = z = typeof y; is a complex assignment statement.
  2. It's evaluated from right to left:
    • typeof y is "undefined" (y hasn't been assigned yet)
    • z is assigned "undefined"
    • y is assigned "undefined"
  3. console.log(y); outputs "undefined".
  4. The final values: y is "undefined", z is "undefined".

Question 75:

What will be the output for below javascript code?

const nums = [1,2,3,4,5,6,7];
nums.forEach((n) => {
    if(n%2 === 0)
      break;
    console.log(n);
});
Click to view output

Output:

SyntaxError: Illegal break statement

Explanation:

This code attempts to use a break statement inside a forEach loop, which is not valid JavaScript. The forEach method doesn't support break to exit the loop early. When the code runs, it will throw a SyntaxError indicating that break is not allowed in this context.

If the intention is to exit the loop early when an even number is encountered, a different loop construct like for...of should be used, or the forEach loop should be replaced with an array method that supports early termination, such as some() or every(). Alternatively, return can be used within the forEach callback to skip to the next iteration, but it won't completely stop the loop.

Question 76:

What will be the output for below javascript code?

function foo() {
  return "I'm the outer function";
}

function test() {
  console.log(bar);
  return foo();
  var bar = "I'm a variable";
  function foo() {
    return "I'm the inner function";
  }
}
console.log(test());
Click to view output

Output:

undefined

I'm the inner function

Explanation:

This code demonstrates function hoisting, variable hoisting, and function scope in JavaScript. When executed, it will output:

undefined I'm the inner function

The outer foo function is never called. The test function contains a local foo function that shadows the outer one. The bar variable is hoisted but not initialized when logged, so it's undefined. The inner foo function is hoisted and fully defined before the return statement, so it's this version that gets called and returned.

Question 77:

What will be the output for below javascript code?

async function foo() {
  console.log('A');
  await Promise.resolve();
  console.log('B');
  await new Promise((resolve) => setTimeout(resolve, 0));
  console.log('C');
}

console.log('D');
foo();
console.log('E');
Click to view output

Output:

D, A, E, B, C

Explanation:

This code demonstrates the asynchronous behavior in JavaScript using async/await and the event loop. The output will be: D, A, E, B, C

The execution flow is as follows:

  1. "D" is logged immediately.
  2. foo() is called, logging "A" synchronously.
  3. The first await creates a microtask, allowing the main thread to continue.
  4. "E" is logged.
  5. The microtask queue is processed, logging "B".
  6. The second await sets a timer, which goes into the macrotask queue.
  7. After the next event loop tick, "C" is logged.

This showcases how asynchronous operations interact with the JavaScript event loop and task queues.

Question 78:

What will be the output for below javascript code?

setTimeout(() => console.log(1), 0);

console.log(2);

new Promise((res) => {
  console.log(3);
  res();
}).then(() => console.log(4));

console.log(5);
Click to view output

Output:

2, 3, 5, 4, 1

Explanation:

This code demonstrates the event loop, microtasks, and macrotasks in JavaScript. The output will be: 2, 3, 5, 4, 1

The execution order is as follows:

  1. The setTimeout callback is scheduled as a macrotask.
  2. "2" is logged synchronously.
  3. The Promise executor function runs synchronously, logging "3".
  4. The Promise's then callback is scheduled as a microtask.
  5. "5" is logged synchronously.
  6. The microtask queue is processed, logging "4".
  7. The call stack empties, allowing the event loop to process the macrotask queue.
  8. The setTimeout callback runs, logging "1".

This illustrates how synchronous code, microtasks (Promises), and macrotasks (setTimeout) are prioritized and executed in JavaScript.

Question 79:

What will be the output for below javascript code?

function getAge() {
  'use strict';
  age = 21;
  console.log(age);
}

getAge();
Click to view output

Output:

ReferenceError: age is not defined

Explanation:

This code demonstrates the effect of strict mode in JavaScript. The output will be: ReferenceError: age is not defined

Key points:

  • The function getAge() uses 'use strict' directive, enabling strict mode.
  • In strict mode, variables must be declared before use.
  • age = 21 attempts to assign a value to an undeclared variable.
  • This results in a ReferenceError, as strict mode prohibits implicit global variable creation.
  • The console.log(age) line is never reached due to the error.

Without 'use strict', the code would create a global age variable and log 21. Strict mode helps catch common coding errors and prevents some unsafe actions.

Question 80:

What will be the output for below javascript code?

console.log(+true);
console.log(-true);
console.log(!'Tanish');
console.log(!!'Tanish');
Click to view output

Output:

1 -1 false true

Explanation:

This code demonstrates type coercion and boolean logic in JavaScript. The output will be: 1 -1 false true

Breaking it down:

  1. +true: The unary plus operator coerces true to number 1.
  2. -true: The unary negation operator coerces true to 1, then negates it to -1.
  3. !'Tanish': The logical NOT operator. Any non-empty string is truthy, so NOT makes it false.
  4. !!'Tanish': Double NOT. First NOT converts the string to false, second NOT converts false to true.

These examples showcase how JavaScript handles type conversion in various contexts, particularly with boolean and string types interacting with numeric and logical operators.

Question 81:

What will be the output for below javascript code?

console.log(typeof typeof 1);
Click to view output

Output:

string

Explanation:

  1. typeof 1 evaluates to "number"
  2. typeof "number" then evaluates to "string"

So, the typeof operator applied twice results in "string".

Contributing

Pull requests are welcome. If you want to add any output based questions that you want to share with others, feel free to do so.

License

MIT