《JavaScript 模式》读书笔记(3)— 字面量和构造函数1

新的篇章开始了,本章开始,所有的内容都是十分有价值和意义的。本章主要的内容包括对象字面量、构造函数、数组字面量、正则字面量、基本值类型字面量以及JSON等。在大家的工作和实际应用中也有一定的指导意义。

大家好,又见面了,我是你们的朋友全栈君。

  新的篇章开始了,本章开始,所有的内容都是十分有价值和意义的。本章主要的内容包括对象字面量、构造函数、数组字面量、正则字面量、基本值类型字面量以及JSON等。在大家的工作和实际应用中也有一定的指导意义。

 

一、对象字面量

  我们直接来先看一下代码:

// 开始时定义一个空对象
var dog = {};

// 向dog对象添加一个属性
dog.name = "Benji";

// 现在,向dog对象添加一个方法
dog.getName = function() {
    return dog.name;
};

// 前面的例子中,dog对象开始是干净的状态,即一个空对象(这里的空对象是指除了继承以外的属性,并没有自身的属性)。
// 然后可以向该对象添加一个属性和一个方法。在程序生命周期的任何时候,都可以执行以下操作。

// 改变属性和方法的值:
dog.getName = function() {
    // 重新定义返回该方法
    // 返回一个硬编码的值
    return "Fido";
};

// 完全删除属性/方法:
delete dog.name;

// 添加更多的属性和方法:
dog.say = function () {
    return "Woof!";
};

dog.fleas = true;

  当然,并不要求必须从空对象开始。对象字面量模式可以使您在创建对象时向其添加函数:

var dog = {
    name:"Benji",
    getName:function() {
        return this.name;
    }
};

  ok,上面的例子,就是一个标准的对象字面量写法,对象字面量语法如下:

    1、将对象包装在打括号中。

    2、对象中以逗号分隔属性和方法。在最后的名称-值的尾随逗号是允许的,但是在一些特殊的浏览器下会报错,所以请尽量不要这样。

    3、用冒号来分隔属性的名称和属性的值。

    4、当给变量赋值时,请不要忘记右大括号“}”后的分号。

 

来自构造函数的对象

  我们先来看一下,如何创建一个对象:

// 第一种方法:使用字面量:
var car = {goes:'far'};

// 另一种方法:使用内置构造函数-这是一个反模式,不要应用!
var car = new Object();
car.goes = "far";

  上面的两种创建对象的方式,尤其注意,不要使用内置构造函数来创建对象。字面量表示发的显著优点在于它仅需要输入更短的字符。优先选择字面量模式创建对象的另一个原因在于:它强调了该对象仅仅是一个可变哈希映射,而不是从对象中提取的属性或方法。

  与使用object构造函数相对,使用字面量的另一个原因在于它并没有作用域解析。因为可能以同样的名字创建了一个局部构造函数,解释器需要从调用Object()的位置开始一直向上查询作用域链,而不是从对象中提取的属性或方法。

 

对象构造函数捕捉

  我们不会使用对象构造函数去创建对象,但是我们应该了解对象构造函数的“特征”。这里的涉及到的“特征”在于,Object()构造函数仅接受一个参数,并且还依赖传递的值,该Object()可能会委派另一个内置构造函数来创建对象,并且返回了一个并非期望的不同对象。

// 以下是反模式

// 一个空对象
var o = new Object();
console.log(o.constructor === Object); //true

// 一个数值对象
var o = new Object(1);
console.log(o.constructor === Number); //true
console.log(o.toFixed(2)); //"1.00"

// 一个字符串对象
var o = new Object('I am a String');
console.log(o.constructor === String);// true

// 一般的对象并没有substring()方法
// 但是字符串对象都有该方法
console.log(typeof o.substring); // "function"

// 一个布尔对象
var o = new Object(true);
console.log(o.constructor === Boolean); //true

  我们看上面的代码,将数字、字符串、布尔值传递到new Object()构造函数中,其结果是获得了不同构造函数所创建的对象。就拿最后一个创建一个Boolean对象来说,实际上,真正执行创建的内置构造函数并不是Object(),而是Object()委派Boolean()构造函数创建的。

  当传递给Object()构造函数的值是动态的,并且直到运行时才能确定其类型时,Object()构造函数的这种行为可能会导致意料不到的结果。因为,最后再强调一遍,不要使用new Object()构造函数!

 

二、自定义构造函数

    除了对象字面量模式和内置的构造函数以外,可以使用自己的构造函数来创建对象:

var adam = new Person("Adam");
adam.say();

  这里的Person,只是一个函数而已:

var Person = function(name) {
    this.name = name;
    this.say = function () {
        return "I am" + this.name;
    };
};

  当使用new操作符调用构造函数时,函数内部会发生以下情况:

    1、创建一个空对象并且this变量引用了该对象,同时还继承了该函数的原型。

    2、属性和方法对加入到this引用的对象中。

    3、新创建的对象由this所引用,并且最后隐式的返回this(如果没有显示的返回其它对象)。

  看起来就像这样:

var Person = function(name) {
    // 使用对象字面量模式创建一个新对象
    // var this = {};
    // 向this添加属性和方法
    this.name = name;
    this.say = function () {
        return "I am" + this.name;
    };

    // return this;
};

  在上面的例子中,为了简单起见,将say()方法添加到了this中。其造成的结果是在任何时候调用new Person()时都会在内存中创建一个新的函数。这种方法的效率显然非常低下,因为多个实例之间的say()方法实际上并没有改变,更好的选择是将方法添加到Person类的原型中。

Person.prototype.say = function () {
    return "I am" + this.name
}

  这里要强调的一点事,可重用的成员,比如可重用的方法,都应该放置到对象的原型中。

  要注意:

// var this = {};

  这段代码并不是真相的全部。因为“空”对象实际上并不空,它已经从Person的原型的原型中继承了许多成员。因此,它更像是下面的语句:

//var this = Object.create(Person.prototype);

  Object.create会在后续的内容中进一步讨论。

 

构造函数的返回值

  当使用new操作符创建对象时,构造函数总是返回一个对象;默认情况下返回的是this所引用的对象。如果在构造函数中并不向this添加任何属性,将返回“空”对象(这里的空,指的是除了从构造函数的原型中所继承的成员以外)。

  构造函数将隐式返回this,甚至于在函数中没有现实的加入return语句。但是,可以根据需要返回任意其他对象。比如:

var Objectmaker = function () {
    // 下面的“name”属性将被忽略
    // 这是因为构造函数决定改为返回另一个对象
    this.name = "This is it";
    // 创建并返回一个新对象
    var that = {};
    that.name = "And that's that";
    return that;
}
// 测试
var o = new Objectmaker();
console.log(o.name);

  正如上面所看到的,可以在构造函数中自由的返回任意对象,只要它是一个对象。试图返回并非对象的值,这虽然不会造成错误,但是函数却会简单的忽略该值,相反,构造函数将会返回this所引用的对象。

 

  好了,我们今天的内容就先到这里。后面再继续,以防一篇的文章内容过长。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/124396.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux运维面试题总结「建议收藏」

    linux运维面试题总结「建议收藏」一、问答题1、安装linux系统对硬盘分区时,必须有那两种分区类型?2、简述raid0、raid1、raid5三种工作原理及特点3、linux下如何改ip,主机名,dns?4、一个ext3的文件分区,当使用touchtest.file命令创建一个新文件时报错,报错的信息是显示磁盘已满,但是采用df-h命令查看磁盘大小时,只使用了60%的磁盘空间,为什么会出现这个情况,说说你的理由5、…

    2022年5月4日
    49
  • HibernateTemplate的常用方法「建议收藏」

    HibernateTemplate的常用方法「建议收藏」HibernateTemplate提供非常多的常用方法来完成基本的操作,比如通常的增加、删除、修改、查询等操作,Spring2.0更增加对命名SQL查询的支持,也增加对分页的支持。大部分情况下,使用Hibernate的常规用法,就可完成大多数DAO对象的CRUD操作。 下面是HibernateTemplate的常用方法简介:     voiddelete(Objecte…

    2022年6月22日
    33
  • html 怎么让整体居中,html中表格整体居中 详解html里面如何让表格居中[通俗易懂]

    html 怎么让整体居中,html中表格整体居中 详解html里面如何让表格居中[通俗易懂]把表格在页面中间显示。。。分享代码。。。在这个无谓的年华,无论别人多么高高不可攀比,但小编还是选择,做一个适应自己的人。首先打开vscode编辑器,新建一个html文档,里面写入一个外层的div,再加入一行table表格:知道谢每一粒种子,每一缕清风,也知道早起播种和御风而行。然后在上方的style标签中加入css样式,设置table标签的样式,table的元素具有长度自适应性,其长度根据其内…

    2022年9月19日
    1
  • 「7年了!GTA 5联机版加载还是这么慢??一个if语句循环了19.8亿次??你的CPU在抽烟」

    「7年了!GTA 5联机版加载还是这么慢??一个if语句循环了19.8亿次??你的CPU在抽烟」你以为我上GitHub就是在学习?你以为我上GTA5就一定是在玩游戏?「7年了!GTA5联机版加载还是这么慢??」别急先献上地址https://github.com/tostercx/GTAO_Booster_PoC详情下面慢慢去了解只需要如下操作:gitclone—recurse-submoduleshttps://github.com/tostercx/GTAO_Booster_PoC之后,把dll文件粘贴到游戏根目录下就OK!彻底提升启动速度70%△Pleasewaitfor

    2022年4月30日
    210
  • python错误和异常处理_python异常处理

    python错误和异常处理_python异常处理抛出异常Python使用raise语句抛出一个指定的异常。raise语法格式如下:raise[Exception[,args[,traceback]]]defdivision():”’功能:分苹果”’print(“\n=====================分苹果了=====================\n”)apple=int(input(“请输入苹果的个数:”))#.

    2022年10月9日
    1
  • oracle隐式转换_oracle查看游标数量

    oracle隐式转换_oracle查看游标数量原文地址:http://blog.itpub.net/29324876/viewspace-1096741/1     Oracle 隐式转换Oracle中对不同类型的处理具有显式类型转换(Explicit)和隐式类型转换(Implicit)两种方式,对于显式类型转换,我们是可控的,但是对于隐式类型转换,当然不建议使用,因为很难控制,有不少缺点,但是我们很难避免

    2022年10月11日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号