js中,变量的数据类型虽然是不确定的,但是各种运算符对数据类型是有要求的。运算子的类型与预期不符的话,就会自动转换类型,如:
减法运算符预期左右两侧的运算子应该是数值,如果不是,就会自动将它们转为数值。
```
'4' - '3' // 1
```
强制转换
强制转换是指,使用Number,String,Boolean,三个函数,手动将各种类型的值,分布转换成数字,字符串,布尔值
1.Number()
// 数值:转换后还是原来的值
Number(324) // 324
// 字符串:如果可以被解析为数值,则转换为相应的数值
Number('324') // 324
// 字符串:如果不可以被解析为数值,返回 NaN
Number('324abc') // NaN
// 空字符串转为0
Number('') // 0
// 布尔值:true 转成 1,false 转成 0
Number(true) // 1
Number(false) // 0
// undefined:转成 NaN
Number(undefined) // NaN
// null:转成0
Number(null) // 0
如果参数是对象的话,都是NaN,除非是包含单个数值的数组
Number({}) // NaN
Number({a: 1}) // NaN
Number([1, 2, 3]) // NaN
Number([5]) // 5
之所以会这样,是因为Number背后的转换规则比较复杂。
第一步,调用对象自身的valueOf方法。如果返回原始类型的值,则直接
对该值使用Number函数,不再进行后续步骤。
第二步,如果valueOf方法返回的还是对象,则改为调用
对象自身的toString方法。如果toString方法返回原始类型的值,
则对该值使用Number函数,不再进行后续步骤。
第三步,如果toString方法返回的是对象,就报错。
Number()要比parseInt()要严格很多,基本上只要有一个字符无法转成数值,整个字符串就会被转成NaN
parseInt('42 cats') // 42
Number('42 cats') // NaN
2.string()
转换规则如下:
-
原始类型值
数值:转为相应的字符串。 字符串:转换后还是原来的值。 布尔值:true转为字符串"true",false转为字符串"false"。 undefined:转为字符串"undefined"。 null:转为字符串"null"。 -
对象
String方法的参数如果是对象,返回一个类型字符串; 如果是数组,返回该数组的字符串形式。 String({a: 1}) // "[object Object]" String([1, 2, 3]) // "1,2,3" String方法背后的转换规则,与Number方法基本相同, 1.先调用对象自身的toString方法。如果返回原始类型的值, 则对该值使用String函数,不再进行以下步骤。 2.如果toString方法返回的是对象,再调用原对象的valueOf方法。 如果valueOf方法返回原始类型的值,则对该值使用String函数,不再进行以下步骤。 3.如果valueOf方法返回的是对象,就报错。 只是互换了valueOf方法和toString方法的执行顺序。
-
Boolean()
它的转换规则相对简单:除了以下五个值的转换结果为false,其他的值全部为true。
undefined null -0或+0 NaN ''(空字符串)
自动转换
遇到以下三种情况时,JavaScript 会自动转换数据类型,即转换是自动完成的,用户不可见。
第一种情况,不同类型的数据互相运算。
123 + 'abc' // "123abc"
第二种情况,对非布尔值类型的数据求布尔值。
if ('abc') {
console.log('hello')
} // "hello"
第三种情况,对非数值类型的值使用一元运算符(即+和-)。
+ {foo: 'bar'} // NaN
- [1, 2, 3] // NaN
自动转换的规则是这样的:预期什么类型的值,就调用该类型的转换函数。比如,某个位置预期为字符串,就调用String函数进行转换。如果该位置即可以是字符串,也可能是数值,那么默认转为数值。
注:除了加法运算符(+)有可能把运算子转为字符串, 其他运算符都会把运算子自动转成数值。
'5' - '2' // 3
'5' * '2' // 10
true - 1 // 0
false - 1 // -1
'1' - 1 // 0
'5' * [] // 0
false / '5' // 0
'abc' - 1 // NaN
null + 1 // 1
undefined + 1 // NaN
!!注意:null转为数值时为0,而undefined转为数值时为NaN。
一元运算符也会把运算子转成数值。
+'abc' // NaN
-'abc' // NaN
+true // 1
-false // 0