基本类型和引用类型的值
ECMAScript变量可能有两种数据类型的值,基本类型值和引用类型值,基本类型值 指的是简单的数据段,而引用类型值指的是那些可能有多个值构成的对象。
基本数据类型的值都按值访问的,我们可以直接对保存在变量中的值进行操作。而引用类型的值是保存在内存中的对象,JavaScript不允许我们直接对内存进行操作,操作对象的值实际上是通过对象的引用做到的,所以,引用类型的值是按引用访问的。
这种说法不严密,当复制保存着对象的某个变量时,操作的是对象的引用。但在为对象添加属性时,操作的实际上是对象。
JavaScript中的字符串不是引用类型
动态的属性
对象的实例可以动态添加属性,而基本类型值不能改变。
1 | var book = new Object(); |
而对于基本数据类型则不可以
1 | var name = "hello"; |
复制变量值
不仅两种类型值得存储方式不一样,在对基本类型值和引用类型值进行复制操作的时候,也存在不同
对基本类型值复制,将会创建一个相同类型的变量,然后在将原变量的值复制给新变量,这时,新变量和原变量两者相互独立。
1 | var num1 = 5; |
此后,两个变量相互独立,对其中一个变量的操作不会影响另一变量
对引用类型值得复制,同样也会将存储在变量对象中的值复制一份放到 为新变量分配的空间中。不同的是,这个值的副本实际上是一个指针,而这个指针指向存储在堆中的一 个对象。复制操作结束后,两个变量实际上将引用同一个对象。因此,改变其中一个变量,就会影响另 一个变量,如下面的例子所示:
1 | var obj1 = new Object(); |
由于是引用类型值,所以两者相互关联
传递参数
ECMAScript 中所有函数的参数都是按值传递的。也就是说,把函数外部的值复制给函数内部的参 数,就和把值从一个变量复制到另一个变量一样。基本类型值的传递如同基本类型变量的复制一样,而 引用类型值的传递,则如同引用类型变量的复制一样。
在向参数传递基本类型的值时,被传递的值会被复制给一个局部变量(即命名参数,或者用 ECMAScript的概念来说,就是
arguments
中的一个元素),比如:
1 | function addTen(num) { |
在向参数传递引用类型的值时,会把 这个值在内存中的地址复制给一个局部变量,因此这个局部变量的变化会反映在函数的外部。比如 :
1 | function setName(obj) { |
这里的结果充分反映了,函数参数的传递为值传递,如果这里是引用传递的话,person
的name
属性应该为Xiaoming
检测类型
通常用来检测引用类型的值,比如,我们想知道这是什么类型的对象
操作符instanceof
,比如:
1 | alert(person instanceof Object); //变量person是Object吗? |
如果变量的给定的应用类型的实例,则返回true
根据规定,所有的引用类型都是Object的实例