Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Can (a== 1 && a ==2 && a==3) ever evaluate to true?

Moderator note: Please resist the urge to edit the code or remove this notice. The pattern of whitespace may be part of the question and therefore should not be tampered with unnecessarily. If you are in the "whitespace is insignificant" camp, you should be able to accept the code as is.

Is it ever possible that (a== 1 && a ==2 && a==3) could evaluate to true in JavaScript?

This is an interview question asked by a major tech company. It happened two weeks back, but I'm still trying to find the answer. I know we never write such code in our day-to-day job, but I'm curious.

like image 764
Dimpu Aravind Buddha Avatar asked Jan 15 '18 20:01

Dimpu Aravind Buddha


People also ask

Can a == 1 && a == 2 && a == 3 ever evaluate to TRUE?

The question is — Can (a==1 && a==2 && a==3) ever evaluate to true? The answer is — Yes.

Why === is false in Javascript?

Because == (and === ) test to see if two objects are the same object and not if they are identical objects.

What is a block statement in Javascript Mcq?

Explanation: A block of statement can be understand as the set of the zero or more statements. In general, a block of statement has common definition "which combines one or a number of statements into a single statement for ease.


2 Answers

If you take advantage of how == works, you could simply create an object with a custom toString (or valueOf) function that changes what it returns each time it is used such that it satisfies all three conditions.

const a = {    i: 1,    toString: function () {      return a.i++;    }  }    if(a == 1 && a == 2 && a == 3) {    console.log('Hello World!');  }

The reason this works is due to the use of the loose equality operator. When using loose equality, if one of the operands is of a different type than the other, the engine will attempt to convert one to the other. In the case of an object on the left and a number on the right, it will attempt to convert the object to a number by first calling valueOf if it is callable, and failing that, it will call toString. I used toString in this case simply because it's what came to mind, valueOf would make more sense. If I instead returned a string from toString, the engine would have then attempted to convert the string to a number giving us the same end result, though with a slightly longer path.

like image 147
Kevin B Avatar answered Sep 22 '22 03:09

Kevin B


I couldn't resist - the other answers are undoubtedly true, but you really can't walk past the following code:

var aᅠ = 1;  var a = 2;  var ᅠa = 3;  if(aᅠ==1 && a== 2 &&ᅠa==3) {      console.log("Why hello there!")  }

Note the weird spacing in the if statement (that I copied from your question). It is the half-width Hangul (that's Korean for those not familiar) which is an Unicode space character that is not interpreted by ECMA script as a space character - this means that it is a valid character for an identifier. Therefore there are three completely different variables, one with the Hangul after the a, one with it before and the last one with just a. Replacing the space with _ for readability, the same code would look like this:

var a_ = 1;  var a = 2;  var _a = 3;  if(a_==1 && a== 2 &&_a==3) {      console.log("Why hello there!")  }

Check out the validation on Mathias' variable name validator. If that weird spacing was actually included in their question, I feel sure that it's a hint for this kind of answer.

Don't do this. Seriously.

Edit: It has come to my attention that (although not allowed to start a variable) the Zero-width joiner and Zero-width non-joiner characters are also permitted in variable names - see Obfuscating JavaScript with zero-width characters - pros and cons?.

This would look like the following:

var a= 1;  var a‍= 2; //one zero-width character  var a‍‍= 3; //two zero-width characters (or you can use the other one)  if(a==1&&a‍==2&&a‍‍==3) {      console.log("Why hello there!")  }
like image 20
Jeff Avatar answered Sep 20 '22 03:09

Jeff