preface
You wish the world, I wish you no bugs. Hello everyone! I’m Lin Dull!
That year I was eighteen years old, pure, kind, ignorant, green, sunshine, lovely…
Now I’m… When it comes to JS type conversions, it’s still…
I thought after toString() and valueOf(), I was the guy who knew you best…
Until I saw in your heart a man named Symbol. ToPrimitive…
This person, who holds the heart of your conversion, can even replace toString() and valueOf() completely if necessary.
I broke down… Frantically, I had to go to Google and Baidu and beg them to tell me how to beat Symbol. ToPrimitive. Finally, they only said one sentence:
"After reading Lin dull of this article come again wave 3 even ok!!"
😂 😂 😂
Sorry, the dog can not change to eat excrement, bah, disposition is difficult to move stubborn stubborn not change, couldn’t help writing a small short film ha 😄.
In fact, I also want to tell you that the last article is the focus, this article is also the focus, so have a good look oh.
There are a lot of people feel that spent so much time and so much energy to see dull these two types of conversion, just to understand this small knowledge point, feel good loss ah, far from brushing a knowledge point of each interview summary covered by a high sense of accomplishment. In fact, I want to say that the interview summary class of the article can help us to check the gaps, but for the JS basic knowledge I think is also very important. Like see Minute brother today in an interview are the regular 2 front companies of journey and summary (tencent, headlines, ali, jingdong) | nuggets technical essay “the article said:
OK👌, play to play, make to make, type conversion to teach you.
Here’s what you can learn by reading:
- rewrite
toString
andvalueOf
Symbol.toPrimitive
- use
= =
Type conversion when comparing +, -, *, /, %
Type conversion of- A few interview questions from big companies
preparation
Before you start reading, I recommend you take a look at the previous article in this series, “How to Fully Understand Conversions from 206 console.log()”; This will help you read this article.
Let’s review the toPrimitive implementation process mentioned earlier:
1. Override toString and valueOf
How much do you know about toPrimititve’s conversion process?
If you feel that the previous examples are not quite illustrative, that is, you still don’t feel that JS really does follow the flow diagram I drew, you can take a look here.
As we mentioned in 6.1, most objects can be used by looking up the toString or valueOf methods on Object.prototype through the prototype chain.
But if I had a toString or valueOf method, I wouldn’t be able to override object. prototype.
You might be able to override the two methods on the prototype chain in this way:
let obj = {
toString () {
return '1'
},
valueOf () {
return 1}}Copy the code
You can even modify the Object. Prototype method directly:
Object.prototype.toString = function () {
return 1
}
var b = {}
console.log(String(b)) / / '1'
Copy the code
(Of course, this affirmation is not recommended, it affects all objects.)
Topic 1.1 a
(Rewrite toString() and valueOf() to determine if our previous toPrimitive process is correct)
We can override toString() and valueOf on an object. If we add console.log(XXX) to the function we override, we can see if the conversion of the object to its original value is going to work the way we want it to.
Like this:
var b = {
toString () {
console.log('toString')
return 1
},
valueOf () {
console.log('valueOf')
return [1.2]}}console.log(Number(b))
Copy the code
Consider the execution result here 🤔️?
ToPrimitive (obj, ‘Number ‘) is used to convert the Number() method.
That is, the valueOf() function is called, the function returns the original value, and then whether toString() should be called or returned.
- Obviously, because of this
valueOf()
Is overwritten, so callvalueOf()
A reference type is then returned[1, 2]
So it will continue to executetoString()
- Which is the execution
[1, 2].toString()
But at this timetoString()
It’s also overwritten and returns the number1
So we don’t have to[1, 2].toString()
The result is, but directly will1
To return.
So at the end of the process, the answer is:
'valueof'
'toString'
1
Copy the code
So it’s pretty clear that the process is really going in the direction that we want it to go.
1.2 the topic 2
If you understood the previous question, let’s take a look here:
var b = {
toString () {
console.log('toString')
return { name: 'b' }
},
valueOf () {
console.log('valueOf')
return [1.2]}}console.log(String(b))
Copy the code
This time I’m using the String() method to convert b to a String.
Also note that the overridden toString() and valueOf are returned reference data types. Can you imagine what the end result will be?
Let’s look at process analysis:
- Performing rewrite
toString()
Method that returns the reference type{name: 'b'}
- Continue to execute the overwritten
valueOf()
Method that returns the reference type[1, 2]
- Wow, very uncomfortable, after two rounds of conversion or reference type, must throw the wrong.
And, yes, the conversion here ultimately fails, because remember in the flowchart, at the end of the process, if it’s not the original value, it throws an exception.
So the result is:
'toString'
'valueOf'
Cannot convert object to primitive value at String
Copy the code
Wonderful 👏, I seem to have seen a complete understanding of the object conversion mechanism of the dawn of original value!!
1.3 the title three
(The difference between arrays ToString)
We all know that when you convert an array to a string, you convert every item in it to a string and then you concatenate it back.
So why is there a “,” concatenation? Will join() be called when toString() is called?
To test my idea 💡, I did an experiment by rewriting the join() method on an array:
var arr = [1.2]
arr['join'] = function () {
let target = [...this]
return target.join('~')}console.log(String(arr))
Copy the code
In the overridden join function, this represents the array arr being called.
Then change the return value to “~” concatenation, the result is:
"1 ~ 2"
Copy the code
That is, it does implicitly call the join method during String(ARr).
But when we overwrite toString(), we don’t care about the overwritten join:
var arr = [1.2]
arr['toString'] = function () {
let target = [...this]
return target.join(The '*')
}
arr['join'] = function () {
let target = [...this]
return target.join('~')}console.log(String(arr)) / / "1 * 2"
Copy the code
As you can see, toString() has a higher priority than join().
Now we can draw another conclusion:
If an object is an array, the default implementation is to use the return value of the join() method as the return value of toString() when converted to a string without overriding its toString() method.
2. Symbol.toPrimitive
When I was proud of myself for understanding toPrimitive, I learned a guy named Symbol. ToPrimitive.
Looking at this guy reminds me of some of the big brothers I’ve seen before: symbol.hasinstance, symbol.tostringtag.
They both have cool tattoos: Symbol, and the previous Big Brother is able to let us do some custom things, I wonder if this guy is what I think, can also help us rewrite toPrimitive 🤔️?
After learning the truth of the matter, I know that I am not stupid, to guess right.
Symbol. ToPrimitive is a better property than toString() and valueOf().
If you override it in an object, the toString() and valueOf() overrides are not even executed.
(Symbol. ToPrimitive is also called @@toprimitive)
Topic 2.1 a
(Basic use of symbol.toprimitive – return a raw value)
You don’t believe what Lin Dull said? Can we get a whole one?
var b = {
toString () {
console.log('toString')
return { name: 'b' }
},
valueOf () {
console.log('valueOf')
return [1.2[]},Symbol.toPrimitive] () {
console.log('symbol')
return '1'}}console.log(String(b))
console.log(Number(b))
Copy the code
In this case, I rewrote all three of the properties I just mentioned, what do you think the result would be?
😄 Remember what Nerdy just said, Symbol. ToPrimitive has the highest priority, so only the contents of it are executed here.
So the result is:
'symbol'
'1'
'symbol'
1
Copy the code
ToPrimitive returns “1”, but the resulting String(b) is still a String and Number(b) is still a Number.
2.2 the topic 2
(Symbol. ToPrimitive returns a reference type, or no value?)
If its return value is a reference type, or if it returns no value at all, will valueOf or toString continue?
The result will not be… Here, I define objects B and C and override the three properties:
var b = {
toString () {
console.log('b.toString')
return { name: 'b' }
},
valueOf () {
console.log('b.valueOf')
return [1.2[]},Symbol.toPrimitive] () {
console.log('b.symbol')}}var c = {
toString () {
console.log('c.toString')
return { name: 'c' }
},
valueOf () {
console.log('c.valueOf')
return [1.2[]},Symbol.toPrimitive] () {
console.log('c.symbol')
return [1.2]}}console.log(String(b))
console.log(String(c))
Copy the code
Execution Result:
'b.symbol'
'undefined'
'c.symbol'
TypeError: Cannot convert object to primitive value
at String
Copy the code
Process analysis:
String(b)
I printed it out in the processb.symbol
, indicating that it was executedSymbol.toPrimitive
Method, but the method does not return a value and does not continue executionvalueOf()
ortoString()
Instead, it returns a string"undefined"
String(c)
It’s printed in the processc.symbol
, butSymbol.toPrimitive
Is an object, but an error is reported.
So from this problem, we can see:
Symbol. ToPrimitive it can be described as a pawn, ten thousand fu mo open, as long as it is in, will not continue to go down, its return result is as the final return result.
ValueOf (); toString(); toString(); toString();
2.3 the title three
(Symbol. ToPrimitive with parameters)
Do you think Symbol. ToPrimitive is just that simple?
No😺, it can accept parameters!!
It takes a string parameter: hint, which indicates the expected type of the original value to be converted to.
The parameter can be one of the following strings:
"number"
"string"
"default"
Well 😺? Lin dumbfounded me, how and before the introduction of toPrimitive so like ah:
toPrimitive(obj, 'number')
toPrimitive(obj, 'string')
Copy the code
ToPrimitive is converted to a Symbol. ToPrimitive is converted to a Symbol.
So cool function, hurry to try:
var b = {
toString () {
console.log('toString')
return '1'
},
valueOf () {
console.log('valueOf')
return [1.2[]},Symbol.toPrimitive] (hint) {
console.log('symbol')
if (hint === 'string') {
console.log('string')
return '1'
}
if (hint === 'number') {
console.log('number')
return 1
}
if (hint === 'default') {
console.log('default')
return 'default'}}}console.log(String(b))
console.log(Number(b))
Copy the code
The toString, valueOf, and symbol. toPrimitive attributes are overwritten. We already know from 👆 above that the first two attributes are ignored as long as symbol. toPrimitive is present, so we do not care about them.
For symbol. toPrimitive, I add all three hint cases. If I had expected String(b) to print String, Number(b) to print Number, and the result is exactly what I expected:
'string'
'1'
'number'
1
Copy the code
So what does “default” do? When was it executed?
At first I thought “default” would be executed if there was no if (hint === ‘string’).
If (hint === ‘string’) and ‘number’ will not execute “default” :
var b = {
toString () {
console.log('toString')
return '1'
},
valueOf () {
console.log('valueOf')
return [1.2[]},Symbol.toPrimitive] (hint) {
console.log('symbol')
// if (hint === 'string') {
// console.log('string')
// return '1'
// }
// if (hint === 'number') {
// console.log('number')
// return 1
// }
if (hint === 'default') {
console.log('default')
return 'default'}}}console.log(String(b))
console.log(Number(b))
// 'symbol'
// 'undefined'
// 'symbol'
// NaN
Copy the code
As you can see, the result is similar to b in problem 2.2, which has no return value.
So, the hint is determined when the Symbol. ToPrimitive is called and will not be changed later.
For example, String(b) is a String, Number(b) is a Number.
In the case of default, which involves the + operator, we’ll see in section 4.
2.4 the title four
Young man (girl), I hear that you have mastered Symbol. ToPrimitive?
OK👌, let’s do a question to consolidate:
class Person {
constructor (name) {
this.name = name
}
[Symbol.toPrimitive] (hint) {
if (hint === 'default') {
console.log('default')
return 'default'
}
if (hint === 'string') {
console.log('string')
return '1'
}
if (hint === 'number') {
console.log('number')
return 1}}}let p1 = new Person('p1');
let p2 = new Person('p2');
console.log(String(p1))
console.log(Number(p2))
console.log(p1)
console.log(p2)
Copy the code
I replaced the original object with the current class. You don’t have to think about it too much. In fact, the instance generated with this object is an object, and can use Symbol.
So the result here is:
'string'
'1'
'number'
1
Person{ name: 'p1' }
Person{ name: 'p2' }
Copy the code
Symbol. ToPrimitive = Symbol. ToPrimitive = Symbol.
Remember that any method defined in a class is defined on its prototype object, person.prototype. Therefore, although P1 and P2 follow Symbol. ToPrimitive, they are used in its prototype chain.
Summary – Symbol. ToPrimitive
So let’s sum it up.
- If you overwrite an object or a constructor
ToString, valueOf, symbol.toprimitive
Method,Symbol.toPrimitive
Is the highest priority - if
Symbol.toPrimitive
An error is reported if the function returns a value that is not the underlying data type (that is, the original value) Symbol.toPrimitive
Accepts a string argumenthint
, which represents the expected type of the original value to be converted to'number', 'string', 'default'
Three options- use
String()
Call,hint
for'string'
; useNumber()
When,hint
for'number'
hint
The value of the parameter is determined from the beginning of the call
To tell the truth, this time is really a bit inflated, now we understand both the implementation mechanism of toPrimitive and the customization of Symbol. ToPrimitive.
3. Use type conversion when comparing ==
Above 👆 whole so many questions, you pour to 👴 point actual meeting test of the thing ah.
Ok, well, in practice, we’re probably going to be quizzed more about using == to compare whether two different types of variables are equal or not.
However, the case of congruence === is relatively simple and generally not very likely to be tested, because the condition of congruence is: if the types are equivalent and equal, they are considered congruent, and type conversion is not involved.
== == == == == ==
console.log([] == ! [])// true
console.log({} == true) // false
console.log({} == "[object Object]") // true
Copy the code
How to? These questions are often seen 😁, let us see one by one below.
First of all, there are a few things we need to know, and this is a hard and fast rule, and we can’t continue without it.
When using == for comparison, there are the following conversion rules (judgment rules) :
- If the two sides have the same type and the same value, they are equal
2 = = 3
Must be forfalse
Of the - Both sides of the comparison are basic data types:
- If one party is
Null, and undefined
, the other party must beNull or undefined
Just fortrue
, that is,null == undefined
fortrue
ornull == null
fortrue
Because theundefined
Derived fromnull
- One party is
String
If it is, it isString
toNumber
Again to compare - One party is
Boolean
If it is, it willBoolean
toNumber
Again to compare
- The side of the comparison has a reference type:
- The reference type will follow
ToNumber
To compare (actually itshint
isdefault
, that is,toPrimitive(obj, 'default')
, butdefault
Conversion rules andnumber
Very similar, specific can see3.10
) - If both are reference types, determine whether they refer to the same object
In some articles, it says:
If one is Object and the other is String, Number, or Symbol, the Object is converted to a String and then compared
3. What is the difference between == and ===? (from “God three -(Suggested collection) native JS soul ask, how many can you catch?
Think of it this way, too, because consider the procedure for toPrimitive(obj, ‘number’) :
- If the input value is a reference data type, call first
valueOf()
methods - if
valueOf()
A method returns a primitive datatype, or continues the call if it does nottoString()
- If the call
toString()
Is returned if the return value of is a primitive data type, otherwise an error is reported.
As you can see, valueOf() is executed first, but a reference type executes a valueOf() method. Except for the date type, it returns itself, that is, after valueOf() is executed, it is still a reference type and itself. So we can omit the valueOf() step and say it executes toString() directly, which would make the problem much faster.
(Although you can omit it, you should know that there is actually such a step, which we will verify in question 3.6.)
In order to facilitate memory, I drew a conversion diagram of the three rules behind, next we just need to press the conversion rules of this diagram to do the problem 😁.
(In order to have a better experience, please be sure to keep this picture in mind)
Topic 3.1 a
(Understand the same type, null, undefined)
Let’s do something simple
console.log(1= =1)
console.log(1= =2)
console.log(null= =0)
console.log(null= =false)
console.log(null= = {})console.log(undefined= =0)
console.log(undefined= =false)
console.log(undefined= = {})console.log(null= =null)
console.log(undefined= =undefined)
console.log(undefined= =null)
Copy the code
Remember the conversion rules at the beginning of the problem 😁.
So the answer here is:
console.log(1= =1) // true
console.log(1= =2) // false
console.log(null= =0) // false
console.log(null= =false) // false
console.log(null= = {})// false
console.log(undefined= =0) // false
console.log(undefined= =false) // false
console.log(undefined= = {})// false
console.log(null= =null) // true
console.log(undefined= =undefined) // true
console.log(undefined= =null) // true
Copy the code
Undefined and null are false, except that they are equal to themselves and each other.
I always thought null == 0 or null == false was true because I might have used! Flag is a way to determine whether a value truly is true, which of course is pretty lax as we go along.)
3.2 the topic 2
(Understand the case where one is String and the other is Number)
In this case, the String is converted to a Number and then compared:
console.log('11'= =11)
console.log('1a'= =11)
console.log('11n'= =11)
console.log('0x11'= =17)
console.log('false'= =0)
console.log('NaN'= =NaN)
Copy the code
There could be a couple of traps here, so watch out.
The answer:
console.log('11'= =11) // true
console.log('1a'= =11) // false
console.log('11n'= =11) // false
console.log('0x11'= =17) // true
console.log('false'= =0) // false
console.log('NaN'= =NaN) // false
Copy the code
'11' = = 11
No problem. The string is converted to a number'1a'
When you turn it into numbersNaN
'11n'
Even after the numbersNaN
You might think of it asbigInt
Type, but notice that this is a string'0x11'
In order to0x
Hexadecimal at the beginning, so after converting to a number17
'false'
It’s a string. It’s notfalse
, so the result is false'NaN'
It’s also a string, but in this case it’s trueNaN
If sofalse
Because theNaN
The bastard is not even equal to himself (i.eNaN===NaN
As the result of thefalse
), only useObject.is(NaN, NaN)
To be judged astrue
)
3.3 the title three
(Understand when one party is Boolean)
In this case, the Boolean is converted to Number for comparison. As we learned from the previous article, the Boolean is converted to Number, which is fairly simple. There are only two cases:
- true => 1
- false => 0
So if there is a Boolean side should be very easy to do…
console.log(true= =1)
console.log(false= =0)
console.log(true= ='1')
console.log(false= ='0')
console.log(true= ='0')
console.log(true= ='false')
console.log(false= =null)
Copy the code
It’s pretty simple:
console.log(true= =1) // true
console.log(false= =0) // true
console.log(true= ='1') // true
console.log(false= ='0') // true
console.log(true= ='0') // false
console.log(true= ='false') // false
console.log(false= =null) // false
Copy the code
- The first two are fine,
true
andfalse
To a number is0
and1
- The third
true
Convert to figure1
And then on the other side is the string1
Depending on rule 3, if one side is a string, the string is converted to a number and then compared, so the result is1 = = 1
The result of theta is thetatrue
- On the fourth and third case,
false
Converted to digital0
After, after"0"
It’s also converted to numbers0
, so the result istrue
- The fifth,
true
Be converted to1
.'0'
Be converted to0
, so the result isfalse
- Sixth,
true
Be converted to1
."false"
Be converted toNaN
, so the result isfalse
- Number seven, well, this is actually just rule one,
null
andfalse
It’s not equal in and of itself.
True == ‘0’ 🤔️?
Because we’ve probably seen code like this:
if ('0') {
console.log('I will be executed.')}Copy the code
Here the contents of the if will be executed, since the string ‘0’ converted to a Boolean is indeed true, then I will always assume that true == ‘0’ is correct.
So note that ‘0’ does convert to true, i.e. :
if (true) {
console.log('I will be executed.')}Copy the code
But in this case it is compared to true, so it follows the “Boolean to number” rule.
So it’s really just a matter of conversion order, true == ‘0’ is the first Boolean conversion.
But don’t think it’s a matter of order 😂, that is, if you change the position of true and ‘0’, the result is the same:
console.log('0'= =true) // false
Copy the code
3.4 the title four
(Where one party is the object)
At the beginning of section 3, Nerdhead said that when a party has an object, it will actually perform a ToNumber operation on the object and then compare it, but since the valueOf() of the object is basically itself, we can say that we omitted this step, but just to make sure that we know, So I’m going to verify that:
var b = {
valueOf: function () {
console.log('b.valueOf')
return '1'
},
toString: function () {
console.log('b.toString')
return '2'}}var c = {
valueOf: function () {
console.log('c.valueOf')
return{}},toString: function () {
console.log('c.toString')
return '2'}}console.log(b == 1)
console.log(c == 2)
Copy the code
In this case, valueOf() of B returns a primitive data type
ValueOf () of c returns a reference type.
So the result is:
'b.valueOf'
true
'c.valueOf'
'c.toString'
true
Copy the code
So we can get this picture:
Let’s do two problems and practice.
3.5 the topic 5
(If one party is a non-array object)
console.log({} == true)
console.log({} == false)
console.log({} == 1)
console.log({} == '1')
console.log({} == 0)
console.log({} == Symbol(1))
console.log({} == null)
console.log({} == {})
Copy the code
Wow, at first glance it feels so much better. It’s… I can’t do that.
You just have to remember that when one of the parties is Object, you can turn that Object into a string and compare it.
And reference type to string remember?
There are two cases of array and non-array, which are roughly:
[] => ''
.['1, 2'] => '1 '
- The non-array case is another one
👆, {} to string is actually “[object object]”
So you can see that the above results are all false.
Among the things that might be harder to understand are:
{} == true
, the conversion process is as follows:
{} = =true
"[object Object]"= =true // Object to string
"[object Object]"= =1 // Boolean to number (rule 4, one side is a Boolean, convert to number)
NaN= =1 // Convert a string to a number (rule 3: Convert a string to a number if one party is a string and the other party is a number)
// The result is false
Copy the code
{} = = 1
, the conversion process is as follows:
{} = =1
"[object Object]"= =1 // Object to string
NaN= =1 // Convert a string to a number (rule 3: Convert a string to a number if one party is a string and the other party is a number)
// The result is false
Copy the code
= = {} {}
The two objects have their own separate heap space, so they are not equal.
3.6 the title six
(If one party is an array)
console.log([] == 0)
console.log([1] = =1)
console.log(['1'] = =1)
console.log([] == 1)
console.log(['1'.'2'] = =1)
console.log(['1'.'2'] = = ['1'.'2'])
console.log([{}, {}] == '[object Object],[object Object]')
console.log([] == true)
console.log([] == Symbol('1'))
Copy the code
What is the meaning of the passage?
console.log([] == 0) [] = =0
' '= =0 // [] An empty array is converted to an empty string
0= =0 // An empty string is converted to a number 0
// true
console.log([1] = =1)
[1] = =1
'1'= =1 // [1] non-empty array with length 1, converted to string '1'
1= =1 // The '1' string is converted to the number 1
// true
console.log(['1'] = =1) // The conversion process is the same as above
// true
console.log([] == 1) [] = =1
' '= =1 // An empty array is converted to a string.
0= =1 // An empty string is converted to a number 0
// false
console.log(['1'.'2'] = =1)
['1'.'2'] = =1
'1, 2,'= =1 // array ['1', '2'] converts to string '1,2'
NaN= =1 // the '1,2' string is converted to a NaN
console.log(['1'.'2'] = = ['1'.'2']) // Reference address is different
// false
console.log([{}, {}] == '[object Object][object Object]') = = [{}, {}]'[object Object][object Object]'
// each item in the [{},{}] array is converted to the string '[object object]' and then concatenated
'[object Object],[object Object]'= ='[object Object],[object Object]'
// true
console.log([] == true) [] = =true[] = =1 // One of the entries is Boolean, so turn the Boolean true to the number 1
' '= =1 // One of the entries is an array, so turn [] to an empty string
0= =1 // The empty string is converted to the number 0
// false
console.log([] == Symbol('1'[] = =))Symbol('1')
' '= =Symbol('1')
// false
Copy the code
3.7 the title seven
(understand! Operator conversion)
When we use! Will actually! The following values are converted to Boolean types for comparison, which is the loose case I mentioned in question 3.1.
And I find that the conversion is not ToNumber(), but directly to booleans, so let’s verify:
var b = {
valueOf: function () {
console.log('b.valueOf')
return '1'
},
toString: function () {
console.log('b.toString')
return '2'}}console.log(! b ==1)
console.log(! b ==0)
Copy the code
The execution result here is:
false
true
Copy the code
As you can see,! B It converts directly to false without going through valueOf or toString
3.8 the title eight
Let’s do a couple more problems:
console.log(!null == !0)
console.log(!undefined == !0)
console.log(!!null= =!!!!!0)
console.log(! {} = = {})console.log(! {} = = [])console.log(! {} = = [0])
Copy the code
The answer:
console.log(!null == !0) // true
console.log(!undefined == !0) // true
console.log(!!null= =!!!!!0) // true
console.log(! {} = = {})// false
console.log(! {} = = [])// true
console.log(! {} = = [0]) // true
Copy the code
As you can see, null and 0, which were not equal just now, have been added respectively. After that, it becomes the same.
The first three output results should be fine, let’s look at the last three:
! {} = = {} :
- The first thing to do is
! {}
After conversion, isfalse
- The equivalent of
false == {}
, one side has a Boolean case that converts booleans to numbers, i.e0 = = {}
- One side has an object and converts the object to a string, i.e
0 == '[object Object]'
- One side has a string that converts the string to a number, i.e
0 == NaN
- So the result is
false
! {} = = [] :
- The first thing to do is
! {}
After conversion, isfalse
- The equivalent of
false == []
, one side has booleans, converts booleans to numbers, i.e0 = = []
. - One side has an object and converts the object to a string, i.e
0 = = '0'
- One side has a string that converts the string to a number, i.e
0 = = 0
- So the result is
true
But!!!! {} == [0] and! {} == [].
3.9 the title nine
Can you understand the first few questions now?
Let’s take a look at it again, this time it must be easy:
var b = {
valueOf() {
console.log('valueOf')
return []
},
toString () {
console.log('toString')
return false}}console.log(! [] = = [])console.log(! [] == b)Copy the code
! [] = = [] :
- First the
! []
To a Boolean type,[]
fortrue
, then! []
isfalse
- then
[]
The conversion to numbers is zero0
.0
withfalse
To compare,false
Also converted to0
, so the result istrue
! [] = = b:
- In the same way,
! []
Into thefalse
b
Will be performedvalueOf
And then executetoString
And so is the returnfalse
- So the result is
true
The answer:
true
'valueOf'
'toString'
true
Copy the code
3.10 the title ten
(Understand the hint parameter of the Symbol. ToPrimitive function of the object when == comparison)
var b = {
[Symbol.toPrimitive] (hint) {
console.log(hint)
if (hint === 'default') {
return 2}}}console.log(b == 2)
console.log(b == '2')
Copy the code
From the above 👆 cases, we can see that objects go through a transformation similar to ToNumber when making == comparisons:
- call
valueOf()
- call
toString()
ToPrimitive will receive “default”, not “number”.
So the answer here is:
'default'
true
'default'
true
Copy the code
3.11 Question 11
(Function conversion when using ==)
A function is also an object, so it can be treated like a normal object when making a == comparison.
But I just want to remind you that when you do a == comparison, be careful whether you are comparing the function itself or the return value of the function.
For example, in this problem:
function f () {
var inner = function () {
return 1
}
inner.valueOf = function () {
console.log('valueOf')
return 2
}
inner.toString = function () {
console.log('toString')
return 3
}
return inner
}
console.log(f() == 1)
console.log(f()() == 1)
Copy the code
f()
Represents theinner
This function, sof() == 1
Quite soinner == 1
So this is where it comes ininner
The type conversion of the function is triggeredinner.valueOf()
To return to2
So the first one isfalse
f()()
Represents theinner()
The return value after the call, i.e1
So this is theta1 = = 1
The comparison, it doesn’t involveinner
The type conversion of the function is not triggeredinner.valueOf()
, so the second istrue
.
Results:
'valueOf'
false
true
Copy the code
Summary – compare using ==
After finishing these eleven questions, I believe you should have a better understanding of the comparison of == than before 😁, let’s sum up a wave.
When using == for comparison, there are the following conversion rules (judgment rules) :
- If the two sides have the same type and the same value, they are equal
2 = = 3
Must be forfalse
Of the - Both sides of the comparison are basic data types:
- If one party is
Null, and undefined
, the other party must beNull or undefined
Just fortrue
, that is,null == undefined
fortrue
ornull == null
fortrue
Because theundefined
Derived fromnull
- One party is
String
If it is, it isString
toNumber
Again to compare - One party is
Boolean
If it is, it willBoolean
toNumber
Again to compare
- The side of the comparison has a reference type:
- The reference type will follow similar
ToNumber
To compare (i.etoPrimitive(obj, 'defalut')
- If both are reference types, determine whether they refer to the same object
When a party has objects, the actual comparison is done after the ToNumber operation is performed, but since valueOf() of the object is basically itself, we can assume that this step is omitted.
Here I paste a flow chart, feel draw quite good, you can compare to have a look:
(photo: segmentfault.com/a/119000001…).
4. +, -, *, /, % type conversion
In addition to the type conversion in the == comparison, other operation symbols are also available.
For example, these five are common in the headlines.
There are two main categories:
-, *, /, %
All four operate by converting both sides of the sign to numbers+
Since it is not only a numeric operator, but also a concatenator of a string, there are two cases:
- If both ends are numbers, do numerical calculations
- If one end is a string, the other end is converted to a string for concatenation
Topic 4.1 a
(Type conversions for four simple operators)
First, we will talk about the conversion of the other four operators except the + sign, because the basic data type should be clear, so I will not explain, here is mainly to say the object operation situation:
var b = {}
console.log(b - '2')
console.log(b * '2')
console.log(b / '2')
console.log(b % '2')
console.log(b - [])
console.log(b - {})
Copy the code
B is an object, and when we do this kind of operation, we convert both ends to numbers, and we know that {} is NaN when we convert to numbers, so the answers are all NaN.
The answer:
NaN
NaN
NaN
NaN
NaN
NaN
Copy the code
4.2 the topic 2
(Actual conversion of the four operators – overriding toString() and valueOf())
Let’s rewrite the toString() and valueOf() methods of the b object in 👆 above. If it followed the ToNumber() transformation, what would the following result be?
var b = {
valueOf () {
console.log('valueOf')
return {}
},
toString () {
console.log('toString')
return 1}}console.log(b - '2')
console.log(b * '2')
console.log(b / '2')
console.log(b % '2')
console.log(b - [])
console.log(b - {})
Copy the code
When calling b, valueOf() is called first, if it returns a primitive datatype, otherwise toString() is called. Obviously, valueOf() still returns a reference type, so toString() is always called.
'valueOf'
'toString'
-1
'valueOf'
'toString'
2
'valueOf'
'toString'
0.5
'valueOf'
'toString'
1
'valueOf'
'toString'
1
'valueOf'
'toString'
NaN
Copy the code
Here are the last two outputs.
B – [] :
b
The output is1
Because the[]
Convert to numbers and we know that it is0
, so the result is1
b - {}
- It’s pretty much the same,
b
for1
.{}
Convert to figureNaN
, so the result isNaN
4.3 the title three
(Actual conversion of the four operators – rewrite symbol.toprimitive)
4.1 What else can we rewrite except toString and valueOf?
Symbol. ToPrimitive: What hint parameters do you think it receives?
var b = {
[Symbol.toPrimitive] (hint) {
if (hint === 'default') {
console.log('default')
return 'default'
}
if (hint === 'number') {
console.log('number')
return 1
}
if (hint === 'string') {
console.log('string')
return '2'}}}console.log(b - '2')
console.log(b * '2')
console.log(b / '2')
console.log(b % '2')
console.log(b - [])
console.log(b - {})
Copy the code
Hint (” number “); hint (” number “); hint (” number “);
'number'
-1
'number'
2
'number'
0.5
'number'
1
'number'
1
'number'
NaN
Copy the code
Yeah ~ Feels easy.
4.4 the title four
(+ sign for object conversion)
+b
Is equivalent to turning into numbers+
If there are values on both sides of the sign, the type of the values on both sides is determined. If both sides are digits, the numeric calculation is performed. If one side is a string, the other side is also converted to a string for concatenation
var b = {}
console.log(+b)
console.log(b + 1)
console.log(1 + b)
console.log(b + ' ')
Copy the code
According to this rule, we can get the answer:
NaN
'[object Object]1'
'1[object Object]'
'[object Object]'
Copy the code
4.5 the topic 5
(the difference between the ‘+’ operator and String())
Symbol. ToPrimitive = Symbol.
var b = {
[Symbol.toPrimitive] (hint) {
if (hint === 'default') {
console.log('default')
return 'I'm default'
}
if (hint === 'number') {
console.log('number')
return 1
}
if (hint === 'string') {
console.log('string')
return '2'}}}console.log(+b)
console.log(b + 1)
console.log(1 + b)
console.log(b + ' ')
console.log(String(b))
Copy the code
Since +b follows the path of converting numbers, its hint must be number.
However, in the case of string concatenation like b + 1, instead of string, default is used.
So you can see the answer is:
var b = {
[Symbol.toPrimitive] (hint) {
if (hint === 'default') {
console.log('default')
return 'I'm default'
}
if (hint === 'number') {
console.log('number')
return 1
}
if (hint === 'string') {
console.log('string')
return '2'}}}console.log(+b) // number
console.log(b + 1) // default
console.log(1 + b) // default
console.log(b + ' ') // default
console.log(String(b)) // string
'number'
1
'default'
'I'm default 1'
'default'
'1 I'm default '
'default'
'I'm default'
'string'
'2'
Copy the code
You can see that the conversion rules for b + 1 and String(b) primes are different
{} + 1
String concatenationhint
fordefault
String({})
whenhint
forstring
4.6 the title six
Since I don’t know the difference between default and number and string at 👆 above, I thought I’d rewrite toString and valueOf() to see what happens.
var b = {
valueOf () {
console.log('valueOf')
return {}
},
toString () {
console.log('toString')
return 1}},console.log(+b) // number
console.log(b + 1) // default
console.log(String(b)) // string
Copy the code
The result is:
'valueOf'
'toString'
1
'valueOf'
'toString'
2
'toString'
'1'
Copy the code
I find that the conversion mode of default is similar to that of number. In both cases, valueOf is executed first, if there is valueOf, then valueOf is executed, and then the return value after valueOf is determined. If it is a reference type, toString is continued. (This is also mentioned in question 3.10)
4.7 the title seven
(Date object data conversion)
As we mentioned earlier, date object conversions are special. (in a reference type call to valueOf())
- Normal object conversion
valueOf
What is returned is itself, which is the reference type - Of the date object
valueOf
Returns the number of milliseconds of a numeric type
var date = new Date(a)console.log(date.valueOf())
console.log(date.toString())
console.log(+date)
console.log(' ' + date)
Copy the code
So we can see that the answer here is:
var date = new Date(a)console.log(date.valueOf()) / / 1585742078284
console.log(date.toString()) // Wed Apr 01 2020 19:54:38 GMT+0800
console.log(+date) / / 1585742078284
console.log(' ' + date) // Wed Apr 01 2020 19:54:38 GMT+0800
Copy the code
+date is converted to a number, so the result is identical to date.valueof ().
But we’ll see that the ” + date here is different from the ” + {} above.
Both are converted to strings, but remember the order in which ‘+ {} was converted? ValueOf () is executed before toString(). Since {}.valueof is itself a reference type, toString() is executed.
Date’s + string concatenation does not follow this conversion rule, but instead calls toString() in preference.
Summary – Operator type conversion
Type conversions for several common operators:
-, *, /, %
All four operate by converting both sides of the sign to numbers+
Since it is not only a numeric operator, but also a concatenator of a string, there are two cases:
- If both ends of the number are numerically calculated (unary plus sign
+b
This is equivalent to converting to numbers.) - If one end is a string, the other end is converted to a string for concatenation
Object + type conversion:
- Object in progress
+
When the string is connected,toPrimitive
The parameters of thehint
isdefault
, butdefault
The execution sequence andnumber
It’s always a matter of judgmentvalueOf
If so, executevalueOf
And then judgevalueof
If the value is a reference type, execution continuestoString
. (similar to the topic4.5
and4.6
) - Date in progress
+
Number strings are called preferentially when connectedtoString()
Methods. (similar to the topic4.7
) - The unary plus sign is the fastest way to convert other objects to values, and it is the most recommended because it does not perform any unnecessary operations on values.
mm…
I wonder if you can see it now? I don’t think it’s fried yet
With so much foreshadowing, it’s time to show the technology in the works!
Now let’s do a few comprehensive questions to check.
5. Several interview questions from big companies
5.1 What is the following output?
console.log([] == [])
console.log([] == ! [])console.log({} == 1)
Copy the code
= =
If both sides of the sign are reference types, it checks whether they are the same reference[] = =! []
This is in3.9
It goes into a lot of detail{} = = 1
In simple terms, convert both sides to numbers,{}
Convert to the numberNaN
, so the result isfalse
. In detail: one side is an object, and the object is converted to a string for comparison, i.e"[object Object]" == 1
; One side has strings that are converted to numbers for comparison, i.eNaN == 1
, so the result isfalse
The answer is:
console.log([] == []) // false
console.log([] == ! [])// true
console.log({} == 1) // false
Copy the code
5.2 What is the following output?
console.log({} + "" * 1)
console.log({} - [])
console.log({} + [])
console.log([2] - [] + function () {})
Copy the code
{} + “” * 1:
- The operation sequence follows multiplication followed by addition, so it is executed first
"" * 1
, the results for0
Because the""
Convert to a number0
- After performing
{} + 0
That will be{}
The conversion to string is"[object Object]"
.0
The conversion to string is"0"
- So the result is
"[object Object]0"
{} – [] :
-
Convert both sides to numbers,{}
forNaN
.[]
for0
, so the result isNaN
{} + [] :
{}
Convert to string"[object Object]"
.[]
Convert to string""
, so the result is"[object Object]"
[2] – [] + function () {} :
-
The two sides of the sign are converted to numbers respectively2
and0
, so[2] - []
The results for2
- after
2 + function () {}
, the two sides are converted to string concatenation as"2function () {}"
Because functions are converted to source strings.
The answer is:
console.log({} + "" * 1) // "[object Object]0"
console.log({} - []) // NaN
console.log({} + []) // "[object Object]"
console.log([2] - [] + function () {}) // "2function () {}"
Copy the code
5.3 How many ways can you make if(a == 1&&a == 2&&a == 3) true?
I’m sure you’ll see a lot of this, but what else can you do besides rewrite valueOf()?
Solution one: rewritevalueOf()
This solution uses: ValueOf () is executed first, if the value returned by valueOf() is a primitive data type. ToString () is called if it is not a reference type. ToString () is returned if it is a primitive data type. Otherwise, an error is reported.
ValueOf () now returns a numeric type at a time, so it returns directly.
/ / 1
var a = {
value: 0,
valueOf () {
return ++this.value
}
}
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
Solution two: RewritevalueOf()
andtoString()
var a = {
value: 0,
valueOf () {
return {}
},
toString () {
return ++this.value
}
}
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
The principle is the same as solution 1, except that toString() is continued when valueOf() returns a reference.
You don’t even have to override valueOf() here, because every other object except the date object returns itself when valueOf() is called.
That means you can also do this:
var a = {
value: 0,
toString () {
return ++this.value
}
}
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
Solution 3: RewriteSymbol.toPrimitive
Symbol. ToPrimitive Symbol.
When the object is compared ==, the hint received by symbol. toPrimitive is “defalut”.
var a = {
value: 0[Symbol.toPrimitive] (hint) {
if (hint === 'default') {
return ++this.value
}
}
}
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
And that’s ok.
Define class and rewrite valueOf()
You can also use class:
class A {
constructor () {
this.value = 0
}
valueOf () {
return ++this.value
}
}
var a = new A()
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
Solution 5: Converting an array to a string is called implicitlyjoin()
What? Is there another way to do it? And I’m a little confused about solution five.
Let’s go back to question 1.3, where we mentioned that when an array is converted to a string, the result of calling toString() is actually the result of calling JOIN.
What does that have to do with this problem? Take a look at the answer:
let a = [1.2.3]
a['join'] = function () {
return this.shift()
}
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
As we know, if an object is an array, the default implementation is to use the return value of the join() method as the return value of toString() when converted to a string without overriding its toString() method.
So here we rewrite a’s join method, and this rewrite does two things:
- The array
a
performa.shift()
Method, which we know affects the original arraya
Phi, let’s get rid of the first term - Let’s go back to the first term that we just removed
So when we perform a == 1, we implicitly call a[‘join’], so we do the two things 👆 said above, a == 2 and a == 3.
Define class to inherit Array and rewrite join()
We can also use class for solution 5
class A extends Array {
join = this.shift
}
var a = new A(1.2.3)
if (a == 1 && a == 2 && a == 3) {
console.log('set up')}Copy the code
This writing method is relatively cool 🆒, but the first time to see may not be able to understand.
- First of all,
A
This class passesextends
Inheritance inArray
And so throughnew A
I’m just creating an array - then
A
Rewrite thejoin
Method,join = this.shift
Is equivalent tojoin = function () { return this.shift() }
- So when every call
a == xxx
“, will implicitly call our customjoin
Method, perform the same operation as method 5.
5.4 Let if (a === 1 &&a === 2 &&a === 3) be true
This one looks a little bit like the one above, but it’s congruent.
We know the conditions for congruence:
- The left and right types must be equal. If the types are not equal, return
false
, and= =
Different,= =
Implicit type conversions occur - The values are not equal
For the solution of the above 👆 problem, we all take advantage of the fact that == will have implicit type conversion. Obviously, if we use it to solve this problem, it cannot be realized.
Consider that when we make a === XXX judgment, we actually call the data a, that is to say, we need to do something before calling the data to achieve our purpose.
I wonder if this makes you think of something 🤔️? You might think of Vue’s famous data hijacking 😁.
Considering that the object.defineProperty () method was used in Vue2. X to redefine all attributes in data, we can also use it to hijack A and change the GET attribute of a.
var value = 1;
Object.defineProperty(window."a", {
get () {
return this.value++; }})if (a === 1 && a === 2 && a === 3) {
console.log('set up')}Copy the code
Here are a few things that actually happen:
- use
Object.defineProperty()
Method hijacks global variableswindow
On the properties of thea
- And every time I call
a
When willvalue
Increment, and returns the increment value
(In fact, I would like to use Proxy to do data hijacking, Proxy window, new Proxy(), but the window object does not seem to have effect…)
Method 2
How to do 😂, one encounters this kind of problem I thought of array again…
var arr = [1.2.3];
Object.defineProperty(window."a", {
get () {
return this.arr.shift()
}
})
if (a === 1 && a === 2 && a === 3) {
console.log('set up')}Copy the code
Poisoned by Shift ()… Of course, this is also possible.
Method three
There is EnoYao big guy there look SAO operation:
Original link: juejin.cn/post/684490…
varA ㅤ =1;
var a = 2;
varㅤ a =3;
if(a ㅤ = =1 && a == 2&& ㅤ = = a3) {
console.log("Set up");
}
Copy the code
Shame to say… I can’t type 😂…
5.5 Implement the following code
Now you need to implement the following functions:
function f () {
/ * * / code
}
console.log(f(1) = =1)
console.log(f(1) (2) = =3)
console.log(f(1) (2) (3) = =6)
Copy the code
When I first saw this problem, IT reminded me of 3.11, except that there were passed parameters and the return value was like a cumulative process.
That is, each parameter passed in is collected and then returned with a sum (this is easy to think of as a Reduce method).
And f of 1 and 2 is written like a partial application, where the function returns a function.
Can we use an array of variables inside function F to hold the set of parameters and then return a function (I’ll call it inner) that collects the parameters passed in and adds them to the set of parameters?
The inner function returned by f will perform implicit type conversion every time the == comparison is performed, that is, the inner valueOf() and toString() methods will be called, so we just need to rewrite these two methods and return the sum of the arguments accumulated by reduce.
The code is also very simple, let’s have a look:
function f () {
let args = [...arguments]
var add = function () { args.push(... arguments)return add
}
add.valueOf = function () {
return args.reduce((cur, pre) = > {
return cur + pre
})
}
return add
}
console.log(f(1) = =1)
console.log(f(1) (2) = =3)
console.log(f(1) (2) (3) = =6)
Copy the code
Of course, the above 👆 valueOf() can be replaced by toString(), because we already know that the order of type conversions for object == comparison is actually valueOf first and then toString.
5.6 What happens when console Input {}+[]?
Let’s not be so solemn… Let’s finish up with an easy and interesting problem.
[]+ {} =? /{} +[] =? (implicit conversion of the plus sign)
(PS: pick a wave of exquisite, the little sister’s articles are quite good, but the heat is not high, we can support it 😁)
OK👌, take a look at the title:
On a console (such as a browser’s console) type:
{} + []Copy the code
What will be the result 🤔️?
The third console.log() in 5.2 has already been done.
console.log({}+[]) // "[object Object]"
Copy the code
But notice that the problem here is to print it on the console.
When I output this code on the console, the answer is different from what I expected:
{} + []0
Copy the code
That is, {} is ignored and +[] is executed, resulting in 0.
ToString () is similar to 1.toString(), because JS interprets the code. In console or terminal, JS will think that the opening of braces {} is an empty code block, so it will ignore it when there is no content in it.
So +[] is just done, converting it to 0.
We don’t have this problem if we change the order:
(Photo credit: juejin.cn/post/684490…)
To verify this, we can call some object methods as if {} were empty and see what happens:
(Console or terminal)
{}.toString()
Copy the code
{} is still considered a block of code rather than an object, so an error is reported:
Uncaught SyntaxError: Unexpected token '. '
Copy the code
The solution is to augment it with a () :
({}).toString
Copy the code
If you want to define the default value of one of the properties as an empty object, you need to use: props
props: {
target: {
type: Object.default: () = >({})}}Copy the code
All right, all right… Ah…
After the language
Knowledge is priceless, support original.
Reference article:
- Uncovering the Type conversion hidden by “==”
- EnoYao: How to set (a===1&& A ===2&&a===3) to true?
- LINGLONG (implicit type conversion for plus)
- Yuba – Heartache of Type Conversion in JavaScript Depth (PART 1)
- God three -(suggested collection) native JS soul of the question, can you catch a few?
You wish the world, I wish you no bugs. So much for this article.
As you can see, it’s not too difficult once we understand how type conversions work. The point is that it’s the part that’s most frightening. As before, I could quickly say that {} == 1 is false, but now my head is doing a double take:
- One side is an object, which is converted to a string for comparison, i.e
"[object Object]" == 1
- One side has strings that are converted to numbers for comparison, i.e
NaN == 1
, so the result isfalse
For some simple questions will even slow a bit… These are not yet ripe caused, I hope you can practice more to avoid this kind of dull state as soon as possible.
Create with your heart and live well. This article after this period of time may not be out of this is the title of the article, do more we will numb do not want to see, so behind some original rational article, please look forward to it, hee hee 😁.
If you find this article helpful, give a thumbs up to 👍, thank you ~ 😁.
The guy who likes Lin Dull also hopes to follow Lin’s public account LinDaiDai or scan the following QR code 👇👇👇.
I will update some front-end knowledge content and my original article 🎉 from time to time
Your encouragement is the main motivation for my continuous creation 😊.
Related recommendations:
The most detailed BPMN.js textbook in the whole Web
If you don’t understand Babel, I’ll send you a mask.
“[Suggested stars] To come to 45 Promise interview questions a cool end (1.1W words carefully arranged)”
“[suggestion 👍] 40 more this interview questions sour cool continue (1.2W word by hand)”
“[why not three even] more simple than inheritance of JS inheritance – packaging (small test)”
【 Why not three times 】 finish these 48 questions to thoroughly understand JS inheritance (1.7W words including hot finishing – back to nature)
Data type conversion from 206 console.log()