包装对象

    所谓“包装对象”,指的是与数值、字符串、布尔值分别相对应的、StringBoolean三个原生对象。这三个原生对象可以把原始类型的值变成(包装成)对象。

    上面代码中,基于原始类型的值,生成了三个对应的包装对象。可以看到,v1v2v3都是对象,且与对应的简单类型值不相等。

    包装对象的设计目的,首先是使得“对象”这种类型可以覆盖 JavaScript 所有的值,整门语言有一个通用的数据模型,其次是使得原始类型的值也有办法调用自己的方法。

    NumberStringBoolean这三个原生对象,如果不作为构造函数调用(即调用时不加new),而是作为普通函数调用,常常用于将任意类型的值转为数值、字符串和布尔值。

    1. // 字符串转为数值
    2. Number('123') // 123
    3. // 数值转为字符串
    4. String(123) // "123"
    5. Boolean(123) // true

    上面这种数据类型的转换,详见《数据类型转换》一节。

    三种包装对象各自提供了许多实例方法,详见后文。这里介绍两种它们共同具有、从Object对象继承的方法:和toString()

    valueOf()方法返回包装对象实例对应的原始类型的值。

    toString()

    toString()方法返回对应的字符串形式。

    1. new Number(123).toString() // "123"
    2. new String('abc').toString() // "abc"
    3. new Boolean(true).toString() // "true"

    某些场合,原始类型的值会自动当作包装对象调用,即调用包装对象的属性和方法。这时,JavaScript 引擎会自动将原始类型的值转为包装对象实例,并在使用后立刻销毁实例。

    比如,字符串可以调用length属性,返回字符串的长度。

    1. var str = 'abc';
    2. str.length // 3
    3. // 等同于
    4. var strObj = new String(str)
    5. // String {
    6. // 0: "a", 1: "b", 2: "c", length: 3, [[PrimitiveValue]]: "abc"
    7. // }

    上面代码中,字符串abc的包装对象提供了多个属性,只是其中之一。

    自动转换生成的包装对象是只读的,无法修改。所以,字符串无法添加新属性。

    上面代码为字符串s添加了一个x属性,结果无效,总是返回undefined

    另一方面,调用结束后,包装对象实例会自动销毁。这意味着,下一次调用字符串的属性时,实际是调用一个新生成的对象,而不是上一次调用时生成的那个对象,所以取不到赋值在上一个对象的属性。如果要为字符串添加属性,只有在它的原型对象String.prototype上定义(参见《面向对象编程》章节)。

    除了原生的实例方法,包装对象还可以自定义方法和属性,供原始类型的值直接调用。

    1. String.prototype.double = function () {
    2. return this.valueOf() + this.valueOf();
    3. };
    4. 'abc'.double()
    5. // abcabc
    6. Number.prototype.double = function () {
    7. return this.valueOf() + this.valueOf();
    8. };
    9. (123).double() // 246

    上面代码在StringNumber这两个对象的原型上面,分别自定义了一个方法,从而可以在所有实例对象上调用。注意,最后一行的123外面必须要加上圆括号,否则后面的点运算符()会被解释成小数点。