这是一个JavaScript问题。
一、valueOf 和 toString
这个问题的一个关键点就是在于利用==
双等号工作原理,==
和===
有什么区别呢?最主要的就是用==
的时候会涉及到类型转换,如果双等号两边数据类型不同会尝试将它们转化为同一类型。基础数据类型之间的转换是比较简单的,这里来说一下对象类型在使用==
时产生的隐形转换。
valueOf 和 toString 这两个方法是每个对象都自带的(继承自 Object 原型),先定义一个简单的对象然后调用他的这两个方法来看下结果:
toString 返回一个字符串"[object Object]"。valueOf 则是直接返回对象本身。而c=="[object Object]"
也为 true,则说明了在隐式转换的过程中,调用了 c 的toString方法。
#二、实现
只要重新定义一下变量 a,重写它的 toString 方法就可以实现我们要达到的目的了:
let a = {
i: 1,
toString () {
return a.i++
}
}
if(a == 1 && a == 2 && a == 3) {
console.log('Hello World!');
}
// 输出Hello World!
具体过程:执行 a1 时,js 引擎会尝试把对象类型 a 转化为数字类型,首先调用 a 的 valueOf 方法来判断,不行则继续调用 toString 方法,然后再把 toString 返回的字符串转化为数字类型再去和 a 作比较(这里我重写了 toString 就直接返回的数字类型的结果,正常情况 toString 返回的字符串)。所以每一次使用判断都会调用一次 a 的 toString 方法,返回i属性的值,然后使 a 的i属性加 1,这样最后的判断结果自然就为 true 了。
其实重写 valueOf 方法也可以实现,而且转化时会优先调用 valueOf 方法:
var c = {
toString () {
console.log('toString')
},
valueOf () {
console.log('valueOf')
}
}
c == 1
// console输出'valueOf'