JavaScript 是一种 弱类型或者说 动态类型语言。因此你不用提早声明变量的类型,在程序运行时,类型会被自动肯定,你也可使用同一个变量保存不一样类型的数据。
<!-- more -->javascript
在 JS 中一共 6 种原始类型:html
除了原始类型其余的都是对象类型(object)了。
在 JS 中,每个数据都须要存放在内存空间中。
原始类型数据直接存储在 栈内存(stack) 中,对象数据存储在 堆内存(heap) 中,并在 栈内存 中存放了该对象数据在 堆内存的地址(引用)。java
var a1 = 0 // 栈 var a2 = 'this is string' // 栈 var a3 = null // 栈 var b = { m: 20 } // 堆 var c = [1,2,3] // 堆
上述变量的内存图解:数组
typeof 是否能正确判断类型?instanceof 能正确判断对象的原理是什么?
typeof
typeof
运算符返回一个字符串,表示未经计算的操做数的类型。
null
类型都显示正确类型。typeof '1' // 'string' typeof true // 'boolean' typeof 1 // 'number' typeof undefined // 'undefined' typeof Symbol() // 'symbol' typeof null // 'object' 这是一个 Bug typeof 1 === 'number' // true typeof (typeof 1) === 'string' // true
object
。typeof [] // 'object' typeof {} // 'object' typeof console.log // `function`
instanceof
instanceof
运算符返回一个布尔值,用于测试构造函数的prototype
属性是否存在于对象 原型链上。因此在判断对象的正确类型时能够考虑使用instanceof
。
// 定义构造函数 function A() {} function B() {} var a = new A() var b = new B() a instanceof A // true ,由于 Object.getPrototypeOf(a) === a.prototype a instanceof B // false ,B.prototype 不在 a 的原型链上
任意数据类型之间能够相互转换,可是 symbol 特殊。symbol 类型只能转换成 string 类型,其余的转换会报错。
Number()
// 1. 字符串 ---> 数字 // 1) 能够被解析成数值的字符串 返回 数值 Number('123') // 123 // 2) 不能解析成数值的字符串 返回 NaN Number('123aaa') // NaN // 3) 空串 ---> 0 Number('') // 0 // 2. 布尔值 ---> 数字 // 1) true ---> 1 Number(true) // 1 // 2) false ---> 0 Number(false) // 0 // 3. undefined ---> NaN Number(undefined) // NaN // 4. null ---> 0 Number(null) // 0
在浏览器环境中,window.parseInt()
和 window.parseFloat()
能够将一些字符串转换成数字,但没 Number()
严格 。把其余原始类型都会转换成 NaN
。浏览器
parseInt('123aaa') // 123 parseFloat('1.23aaa') // 1.23 parseInt(true) // NaN
Number()
方法的参数是对象时,通常状况下,除非是包含单个数字的数组会返回数字,不然返回 NaN
。函数
Number({a: 1}) // NaN Number([1,2]) // NaN Number([1]) // 1
在执行 Number(对象)
方法时,会先进行参数处理,将对象转换成原始类型,再进行转换成数字的操做。测试
valueOf()
方法( var a = new String(123); a.valueOf() --> '123' )。若是返回的值是原始类型,则直接将该值转换成数字并返回。再也不进行后续步骤。valueOf()
方法返回的是对象,则调用原对象自身的 toString()
方法,若是此时返回的值是原始类型,则将该值转换成数字并返回,再也不进行后续操做。toString()
方法返回的仍是对象,就报错。Number({a: 1}) // NaN // 过程以下: var t = {a: 1} // 第一步 t.valueOf() // {a: 1} // 第二步 t.toString() // '[object Object]' Number('[object Object]') // NaN Number([1,2]) //过程以下: // 第一步 [1,2].valueOf() // [1,2] // 第二步 [1].toString() // '1,2' Number('1,2') // NaN // 再来看看参数是单个数字的数组时的状况 Number([1]) // 1 //过程以下: // 第一步 [1].valueOf() // [1] // 第二步 [1].toString() // '1' Number('1') // 1
String()
String()
函数能够将任意类型的值转换成字符串。
var a = Symbol({}) String(a) // 'Symbol([object Object])' String(Symbol([])) // 'Symbol()'
数组会返回该数组的字符串形式,函数返回完整函数的字符串,其余的返回一个类型字符串。this
String([1,2]) // '1,2' String([]) // '' function foo(x){ return x*x } String(foo) // 'function foo(x){ return x*x }' String({a: 1}) // '[object Object]'
String(对象)
方法的转换规则与 Number(对象)
基本相同,只是互换了 valueOf()
与 toString()
的执行顺序。spa
toString()
方法。若是返回的值是原始类型,则将该值转换成字符串并返回。再也不执行如下步骤。toString()
返回的是对象,则调用原对象的 valueOf()
方法。若是返回的值是原始类型,将该值转换成字符串并返回。再也不执行后续操做。valueOf
() 方法返回的仍是对象,就报错。自定义对象的 toString() 方法和 valueOf() 方法:prototype
var obj = { valueOf: function() { return 1 }, toString: function() { return 2 } } String(obj) // '2' Number(obj) // 1
Boolean()
Boolean()
函数能够将任意类型转换成布尔值
除了如下五个值转换结果为false
,其余值所有转换为true
undefined
null
''
(空字符串)-0
或 0
NaN
!!数据
与Boolean(数据)
效果同样。
Boolean(undefined) // false Boolean(null) // false Boolean('') // false Boolean(0) // false Boolean(NaN) // false ! NaN // true !! NaN // false
自动转换就是没有显式地使用函数对数据进行类型转换。但它是以强制转换为基础的。
当预期数值与实际数值不匹配时,JavaScript 就会自动调用预期类型的转换函数对实际值进行转换。
运算子都是原始类型
-
*
/
**
%
++
--
+
(一元运算符 表示正数) -
(一元运算符 表负数) 都会将运算子转换成数字。1 + '1' // '11' 2 * '3' // 6 1 - 'a' // NaN 3 + - '1' // 2
会先把对象转换成原始类型再按预期值转换。具体方法是:先调用对象的 valueOf()
方法试图将对象转换成原始类型,若是返回的是对象,就再调用原对象的 toString()
方法,若是返回的仍是对象就报错。
var obj = { valueOf: function() { return '1' }, toString: function() { return '10' } } 10 - obj // 9 1 + [1,2] // '11,2' // [1,2].toString() ---> '1,2'
===
和 !==
不会发生类型转换,只要类型不一样就返回 false
.NaN
与任意值(包括自己NaN
)比较都返回 false
null
undefined
除了 null(undefined) == undefined(null)
返回 true
,其余 >
<
==
有 null
或 undefined
参与的都返回 false
。'abc' > 'abd' // false true > false // true ---> Number(true) > Number(false) true > '-1' // true ---> Number(true) > Number('-1') var obj = {valueOf: function() {return '1'}} [3] > obj // true // 过程以下: // Number([3]) > Number(obj) // Number('3') > Number('1') // 3 > 1 0 == null // false
==
经常使用于判断函数的参数是否等于 null
或者 undefined
,由于 null == undefinded
返回 true
function fun(x) { if(x == null) {···} }
!参数
参数&&
参数||
参数? :
(三元运算符)if(参数)
都会将非布尔的参数转换成布尔类型(使用 Boolean(参数)
函数)。
参考资料:
https://wangdoc.com/javascript/features/conversion.html
https://juejin.im/book/5bdc715fe51d454e755f75ef/section/5bdc715f6fb9a049c15ea4e0