典型的面向对象编程语言(比如C++和Java),存在“类”(class)这个概念。所谓“类”就是对象的模板,对象就是“类”的实例。但是,在JavaScript语言的对象体系,不是基于“类”的,而是基于构造函数(constructor)和原型链(prototype)。

以下的内容会分为如下细节:

1.对象的概念

2.构造函数

3.new 命令

  3.1:基本原理

  3.2:基本用法

1.对象的概念

  “面向对象编程”(Object Oriented Programming,缩写为OOP)是目前主流的编程范式。它的核心思想是将真实世界中各种复杂的关系,抽象为一个个对象,然后由对象之间的分工与合作,完成对真实世界的模拟。

  在Javascript的重要数据类型-对象这篇文章中,提到了js中对象的一些基本知识,比如说对象的创建,对象的引用,对象的属性等等。如果没有掌握对象的基础知识,请移步Javascript的重要数据类型-对象

  在有了对象的基础知识之后,对js中对象做一个简单的总结。如下:

  a:对象是单个实物的抽象。

  b:对象是一个容器,封装了‘属性’和‘方法’。

  一本书、一辆汽车、一个人都可以是“对象”。当实物被抽象成“对象”,实物之间的关系就变成了“对象”之间的关系,从而就可以模拟现实情况,针对“对象”进行编程。

  所谓属性,就是对象的一种状态;所谓方法,就是对象的一种行为。

  比如说,可以把动物抽象为animal对象,属性记录的就是哪一种动物,以及该动物的大小和颜色等。方法表示该动物的某种行为(奔跑,猎食,交配,休息等等)。

2.构造函数

  ‘面向对象编程’的第一步,就是要生成对象。而js中面向对象编程是基于构造函数(constructor)和原型链(prototype)的。

  前面说过,“对象”是单个实物的抽象。通常需要一个模板,表示某一类实物的共同特征,然后“对象”根据这个模板生成。

  js语言中使用构造函数(constructor)作为对象的模板。所谓构造函数,就是提供一个生成对象的模板,并描述对象的基本结构的函数。一个构造函数,可以生成多个对象,每个对象都有相同的结构。

  看一下构造函数的基本结构。

     var Keith = function() {
         this.height = 180;
     };
     //两种写法相同。
     function Keith() {
         this.height = 180;
     }

  上面代码中,Keith就是构造函数,它提供模板,用来生成对象实例。为了与普通函数区别,构造函数名字的第一个字母通常大写。

  构造函数的三大特点:

  a:构造函数的函数名的第一个字母通常大写。

  b:函数体内使用this关键字,代表所要生成的对象实例。

  c:生成对象的时候,必须使用new命令来调用构造函数。

3.new 命令

  3.1:基本原理

  new命令的作用,就是执行一个构造函数,并且返回一个对象实例。使用new命令时,它后面的函数调用就不是正常的调用,而是依次执行下面的步骤。

  a:创建一个空对象,作为将要返回的对象实例。

  b:将空对象的原型指向了构造函数的prototype属性。

  c:将空对象赋值给构造函数内部的this关键字。

  d:开始执行构造函数内部的代码。

  也就是说,构造函数内部,this指向的是一个新生成的空对象,所有针对this的操作,都会发生在这个空对象上。构造函数之所谓构造函数,意思是这个函数的目的就是操作一个空对象(即this对象),将其构造为需要的样子。

  以上是new命令的基本原理,这个很重要。以下会用具体实例来验证该原理的过程。

  

  3.2:基本用法

  new命令的作用,就是调用一个构造函数,并且返回一个对象实例。

     function Keith() {
         this.height = 180;
     }

     var boy = new Keith();
     console.log(boy.height);  //180

  上面代码中通过new命令,让构造函数Keith生成一个对象实例,并赋值给全局变量boy。这个新生成的对象实例,从构造函数Keith中继承了height属性。也就说明了这个对象实例是没有height属性的。在new命令执行时,就代表了新生成的对象实例boy。this.height表示对象实例有一个height属性,它的值是180。

  使用new命令时,根据需要,构造函数也可以接受参数。

     function Person(name, height) {
         this.name = name;
         this.height = height;
     }

     var boy = new Person('Keith', 180);
     console.log(boy.name); //'Keith'
     console.log(boy.height); 

     var girl = new Person('Samsara', 160);
     console.log(girl.name); //'Samsara'
     console.log(girl.height); 

  用以上的一个例子,来对构造函数的特点和new基本原理进行一个梳理。

  上面代码中,首先,我们创建了一个构造函数Person,传入了两个参数name和height。构造函数Person内部使用了this关键字来指向将要生成的对象实例。

  然后,我们使用new命令来创建了两个对象实例boy和girl。

  当我们使用new来调用构造函数时,new命令会创建一个空对象boy,作为将要返回的实例对象。接着,这个空对象的原型会指向构造函数Person的prototype属性。也就是boy.__proto__ prototype===Person.prototype的。要注意的是空对象指向构造函数Person的prototype属性,而不是指向构造函数本身。然后,我们将这个空对象赋值给构造函数内部的this关键字。也就是说,让构造函数内部的this关键字指向一个对象实例。最后,开始执行构造函数内部代码。

  因为对象实例boy和girl是没有name和height属性的,所以对象实例中的两个属性都是继承自构造函数Person中的。这也就说明了构造函数是生成对象的函数,是给对象提供模板的函数。

  

  一个问题,如果我们忘记使用new命令来调用构造函数,直接调用构造函数了,会发生什么?

  这种情况下,构造函数就变成了普通函数,并不会生成实例对象。而且由于后面会说到的原因,this这时代表全局对象,将造成一些意想不到的结果。

     function Keith() {
         this.height = 180;
     }

     var person = Keith();
     console.log(person.height); //TypeError: person is undefined
     console.log(window.height); 

  上面代码中,当在调用构造函数Keith时,忘记加上new命令。结果是this指向了全局作用域,height也就变成了全局变量。而变量person变成了undefined。

  因此,应该非常小心,避免出现不使用new命令、直接调用构造函数的情况。

  为了保证构造函数必须与new命令一起使用,一个解决办法是,在构造函数内部使用严格模式,即第一行加上use strict

     function Person(name, height) {
         'use strict';
         this.name = name;
         this.height = height;
     }
     var boy = Person();
     console.log(boy) //TypeError: name is undefined

  上面代码的Person为构造函数,use strict命令保证了该函数在严格模式下运行。由于在严格模式中,函数内部的this不能指向全局对象。如果指向了全局,this默认等于undefined,导致不加new调用会报错(JavaScript不允许对undefined添加属性)。

  另一个解决办法,是在构造函数内部判断是否使用new命令,如果发现没有使用,则直接返回一个实例对象。

     function Person(name, height) {
     if (!(this instanceof Person)) {
             return new Person(name, height);
         }
         this.name = name;
         this.height = height;
     }
     var boy = Person('Keith');
     console.log(boy.name) //'Keith'

  上面代码中的构造函数,不管加不加new命令,都会得到同样的结果。

  如果构造函数内部有return语句,而且return后面跟着一个复杂数据类型(对象,数组等),new命令会返回return语句指定的对象;如果return语句后面跟着一个简单数据类型(字符串,布尔值,数字等),则会忽略return语句,返回this对象。

     function Keith() {
         this.height = 180;
         return {
             height: 200
         };
     }
     var boy = new Keith();
     console.log(boy.height); 

     function Keith() {
         this.height = 100;
         return 200;
     }
     var boy = new Keith();
     console.log(boy.height); 

  另一方面,如果对普通函数(内部没有this关键字的函数)使用new命令,则会返回一个空对象。

     function Keith() {
         return 'this is a message';
     }
     var boy = new Keith();
     console.log(boy); // Keith {}

  上面代码中,对普通函数Keith使用new命令,会创建一个空对象。这是因为new命令总是返回一个对象,要么是实例对象,要么是return语句指定的对象或数组。本例中,return语句返回的是字符串,所以new命令就忽略了该语句。

  

完。

感谢大家的阅读。

转载请注明出处:http://www.cnblogs.com/Uncle-Keith/p/5803551.html

Javascript中构造函数与new命令的更多相关文章

  1. Javascript中构造函数与new命令2

    典型的面向对象编程语言(比如C++和Java),存在"类"(class)这个概念.所谓"类"就是对象的模板,对象就是"类"的实例.但是,在J ...

  2. 深入理解Javascript中构造函数和原型对象的区别

    在 Javascript中prototype属性的详解 这篇文章中,详细介绍了构造函数的缺点以及原型(prototype),原型链(prototype chain),构造函数(constructor) ...

  3. Javascript中构造函数的返回值问题和new对象的过程

    首先明确一点:javascript中构造函数是不需要有返回值的,这一点跟java很类似.可以认为构造函数和普通函数的最大差别就是:构造函数中没有return语句,普通函数可以有return语句:构造函 ...

  4. 解决JavaScript中构造函数浪费内存的问题!

    解决JavaScript中构造函数浪费内存的问题! 把构造函数中的公共的方法放到构造函数的原型对象上! // 构造函数的问题! function Gouzaohanshu(name, age, gen ...

  5. 深入理解Javascript中构造函数和原型对象的区别(转存)

    Object是构造函数,而Object.prototype是构造函数的原型对象.构造函数自身的属性和方法无法被共享,而原型对象的属性和方法可以被所有实例对象所共享. 首先,我们知道,构造函数是生成对象 ...

  6. JavaScript中构造函数

    构造函数:函数的另一种执行方法,执行后创建对象,并创建原型对象. 原型链:对象访问构造函数的指针. Function函数:函数对象. Object函数:所有创建对象的祖辈对象,也是由Function对 ...

  7. javascript中构造函数与普通函数的区别还有关于“new”操作符的一些原理

    有一种创建对象的方法叫做工厂模式,例如: function person(name,age){ var o=new Object(); o.name=name; o.age=age; return o ...

  8. javascript中构造函数的三种方式

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. javascript中构造函数知识总结

    构造函数的说明 1.1 构造函数是一个模板 构造函数,是一种函数,主要用来在创建对象时对 对象 进行初始化(即为对象成员变量赋初始值),并且总是与new运算符一起使用. 1.2 new 运算符 new ...

随机推荐

  1. ASP.NET Core 1.0 静态文件、路由、自定义中间件、身份验证简介

    概述 ASP.NET Core 1.0是ASP.NET的一个重要的重新设计. 例如,在ASP.NET Core中,使用Middleware编写请求管道. ASP.NET Core中间件对HttpCon ...

  2. CATransition转场动画

    背景: 最近在温习动画,分享个简单系统的转场动画 viewcontroller *VC=[self.storyboard instantiateViewControllerWithIdentifier ...

  3. YY一下淘宝商品模型

    淘宝的电商产品种类非常丰富,必然得力于其商品模型的高度通用性和扩展性. 下面我将亲自操作淘宝商品的发布过程,结合网上其他博客对淘宝网商品库的分析,简单谈谈我的理解. 注:下面不特殊说明,各个表除主键外 ...

  4. jQuery动画的实现

    没有引入deferred机制,其余流程都有了 //////////// //创建动画缓动对象 // //////////// function Tween(value, prop, animation ...

  5. JavaScript知识 一、JS的数据类型

    一.JS的数据类型 1.基本类型 JS共有5大基本类型,分别是: 1)Undefined.他只有一个值:undefined.如果一个变量被定义但是没有给他赋值,那么这个时候系统会默认给这个变量赋值为u ...

  6. EntityFramework 7 更名为EntityFramework Core(预发布状态)

    前言 最近很少去学习和探索新的东西,尤其是之前一直比较关注的EF领域,本身不太懒,但是苦于环境比较影响自身的心情,所以迟迟没有下笔,但是不去学习感觉在精神层面缺少点什么,同时也有园友说EF又更新了,要 ...

  7. Util应用程序框架公共操作类(二):数据类型转换公共操作类(源码篇)

    上一篇介绍了数据类型转换的一些情况,可以看出,如果不进行封装,有可能导致比较混乱的代码.本文通过TDD方式把数据类型转换公共操作类开发出来,并提供源码下载. 我们在 应用程序框架实战十一:创建VS解决 ...

  8. BAT及各大互联网公司2014前端笔试面试题--Html,Css篇

    很多面试题是我自己面试BAT亲身经历碰到的.整理分享出来希望更多的前端er共同进步吧,不仅适用于求职者,对于巩固复习前端基础更是大有裨益. 而更多的题目是我一路以来收集的,也有往年的,答案不确保一定正 ...

  9. Ajax JQuery HTML 提交上传文件File HTML+ Ajax+ASP.NET+ WebService

    起因:公司最近有些项目用到了HTML+WebService的组合,发现访问速度等都快了许多,但是由于通过Ajax只能应付一些简单的文字类的传输,上传文件就捉襟见肘了,如果一直引用第三方的swf之类上传 ...

  10. [OpenCV] Samples 13: opencv_version

    cv::CommandLineParser的使用. I suppose CommandLineParser::has("something") should be true whe ...