0%

JS---基本类型和引用类型的值

基本类型和引用类型的值

ECMAScript变量可能有两种数据类型的值,基本类型值和引用类型值,基本类型值 指的是简单的数据段,而引用类型值指的是那些可能有多个值构成的对象。

基本数据类型的值都按值访问的,我们可以直接对保存在变量中的值进行操作。而引用类型的值是保存在内存中的对象,JavaScript不允许我们直接对内存进行操作,操作对象的值实际上是通过对象的引用做到的,所以,引用类型的值是按引用访问的。

这种说法不严密,当复制保存着对象的某个变量时,操作的是对象的引用。但在为对象添加属性时,操作的实际上是对象。

JavaScript中的字符串不是引用类型

动态的属性

对象的实例可以动态添加属性,而基本类型值不能改变。

1
2
3
var book = new Object();
book.name = "JavaScript高级程序设计语言"; //为对象book动态添加属性
alert(book.name); //调用book的name属性

而对于基本数据类型则不可以

1
2
3
var name = "hello";
name.age = 23;
alert(name.age) //undefine

复制变量值

不仅两种类型值得存储方式不一样,在对基本类型值和引用类型值进行复制操作的时候,也存在不同

对基本类型值复制,将会创建一个相同类型的变量,然后在将原变量的值复制给新变量,这时,新变量和原变量两者相互独立。

1
2
var num1 = 5;
var num2 = num1;

此后,两个变量相互独立,对其中一个变量的操作不会影响另一变量

对引用类型值得复制,同样也会将存储在变量对象中的值复制一份放到 为新变量分配的空间中。不同的是,这个值的副本实际上是一个指针,而这个指针指向存储在堆中的一 个对象。复制操作结束后,两个变量实际上将引用同一个对象。因此,改变其中一个变量,就会影响另 一个变量,如下面的例子所示:

1
2
3
4
var obj1 = new Object();
var obj2 = obj1;
obj1.name = "Nicholas";
alert(obj2.name); //"Nicholas"

由于是引用类型值,所以两者相互关联

传递参数

ECMAScript 中所有函数的参数都是按值传递的。也就是说,把函数外部的值复制给函数内部的参 数,就和把值从一个变量复制到另一个变量一样。基本类型值的传递如同基本类型变量的复制一样,而 引用类型值的传递,则如同引用类型变量的复制一样。

在向参数传递基本类型的值时,被传递的值会被复制给一个局部变量(即命名参数,或者用 ECMAScript的概念来说,就是arguments中的一个元素),比如:

1
2
3
4
5
6
7
8
9
function addTen(num) {
num += 10;
return num;
}

var num = 20;
addTen(num);
alert(num); // 20
alert(addTen(num)) // 30

在向参数传递引用类型的值时,会把 这个值在内存中的地址复制给一个局部变量,因此这个局部变量的变化会反映在函数的外部。比如 :

1
2
3
4
5
6
7
8
9
function setName(obj) {
obj.name = "Xiaohua";
obj = new Object(); //将局部变量obj重新定义了一个对象
obj.name = "Xiaoming"; //并将name属性改为"Xiaoming"
}

var person = new Object();
setName(person);
alert(person.name); //Xiaohua

这里的结果充分反映了,函数参数的传递为值传递,如果这里是引用传递的话,personname属性应该为Xiaoming

检测类型

通常用来检测引用类型的值,比如,我们想知道这是什么类型的对象

操作符instanceof ,比如:

1
2
alert(person instanceof Object);   //变量person是Object吗?
alert(person instanceof Array); //变量person是Array吗?

如果变量的给定的应用类型的实例,则返回true

根据规定,所有的引用类型都是Object的实例