文章参考:
What the f*ck JavaScript?
再谈:JavaScript 中的对象是如何进行类型转换的?
通常发生在比较运算(==
、!=
、>
、<
)、算术运算(+
、-
、*
、/
、%
)和if语句
,而且运算符两边的操作数不是同一类型。
注: 相等运算符
==
中有特殊情况,null == undefined 的比较结果返回 true。
Number
、String
、Boolean
这都是显示转化
在使用 Number()
或 toNumber()
方法将一个字符串转换为数字时,如果字符串中出现非数字字符,则会返回 NaN
基础数据类型均会转为字符串
只有以下几种情况会转为false
toPrimitive(primitive:原始的),作用:转为基础数据类型。
对象发生类型转换时,首先会检查对象上是否存在 [Symbol.toPrimitive]
属性,如果存在的话就调用。该属性存在一些限制:它必须是一个函数,而且返回值必须是基本类型,否则就要报错
:
例如:
[Symbol.toPrimitive]
在调用时,系统会自动给与一个参数 hint,这个 hint 可以理解为,此次对象发生转换的预期类型为何。
hint 有三种可能取值:string
、number
、default
。
下面我们来演示下,这三种情况的发生场景。
【运行结果】
【代码片段】
var obj = {
[Symbol.toPrimitive](hint) {
console.log('[Symbol.toPrimitive]', hint)
}
}
undefined
// 第一种情况(期望类型是 number):
+obj
// [Symbol.toPrimitive] number
// NaN
// 第二种情况(期望类型是 string):
String(obj)
// [Symbol.toPrimitive] string
// "undefined"
// 第三种情况(未知):
obj + ':('
// [Symbol.toPrimitive] default
// "undefined:("
obj + 1
// [Symbol.toPrimitive] default
// NaN
【结论解释】
[Symbol.toPrimitive]
方法里我们没有定义返回值,因此方法返回值是默认的 undefined
。
第一种情况:+obj -> +undefined -> Number(undefined) -> NaN。
第二种情况:String(obj) -> String(undefined) -> “undefined”。
第三种情况:
然而当对象不存在 [Symbol.toPrimitive]
的时候,转换规则又是怎样的呢?
这就关系到 Object.prototype
对象上的两个方法了:valueOf
和 toString
。
当对象上不存在 [Symbol.toPrimitive]
属性的时候,若发生类型转换,就要用到 Object.prototype
对象上的 valueOf
和 toString
两个方法了。
两个方法调用有先后,可能都会调用,也可以只调用一个就完成转换,返回结果。这跟方法返回值和hint
值有关系。
具体是:
你可能要问了,如果两个方法的返回值都是对象的话,岂不是得不到对象最终的转换结果了?一点都没错,我们来试一下:
上图里,我们在对象 obj 上定义了 valueOf 和 toString 方法,在发生类型转换时,覆盖掉原型对象上的同名方法,以便我们能更加真切地感受到对象内部的实际的转换流程。
我们制造了一个极端情况,两个方法都没有返回基本类型只而是对象,结果呢?然后控制台就报错了,告诉我们不要这样玩。
需要说明的是——在 JavaScript 中,当我们在一个对象上调用 valueOf
方法的时候,实际上调用的是 Object.prototype.valueOf
这个原型方法。默认这个方法的返回值始终是调用对象自身
,也就是说 valueOf
方法的返回的始终是对象,而非一个基本类型值。
注:
1、Date 对象除外,因为 Date.prototype 上定义的 valueOf 方法覆盖掉了 Object.protortype 上的。在 Date 对象上调用 valueOf方法,返回的是时间对象内部的时间戳表示。
2、另外,Date.prototype 也定义了自己的 [Symbol.toPrimitive] 属性,默认是不可写入的。
这就得到了一个结论:对象发生到基本类型转换时,最终转换结果就是 obj.toString() 返回值!这就解释了下面代码里的输出结果:
var obj = {}
// (1)
obj + ' :)' // "[object Object] :)"
// (2)
obj - 1 // NaN
【结果分析】
obj + ’ : )’ -> ‘[object Object]’ + : )’ -> ‘[object Object] : )’
obj - 1 -> ‘[object Object]’ - 1 -> Number(‘[object Object]’) - 1 -> NaN - 1 -> NaN
数组本质也是对象,因此上面的转换规则也适应于它。不同的是数组原型对象上定义了自己的实现方法:Array.prototype.toString
,这个方法覆盖掉了 Object.prototype
上的同名方法,toString
会以,
去拼接显示。
Array.prototype.toString = function () { return this.join() }
因为并未覆盖 valueOf
方法(也是返回对象本身,对应到这里就是数组实例),因此数组转换结果就是调用 array.join()
的返回结果。
下面举几个例子:
var emptyArray = []
var array1 = [123]
var array2 = ['hi', 'How are you']
// (1)
Number(emptyArray) // 0
// (2)
array1 - 100 // 23
// (3)
array2 + '?' // "hi,How are you?"
对应的转换过程如下:
总结下来,一个对象转换到基本类型的算法步骤如下:
到这里的话,对象类型转换的内容就讲完了,希望能帮助到大家!