JavaScript里,构造函数通常是认为用来实现实例的,JavaScript没有类的概念,但是有特殊的构造函数。通过new关键字来调用定义的否早函数,你可以告诉JavaScript你要创建一个新对象并且新对象的成员声明都是构造函数里定义的。在构造函数内部,this关键字引用的是新创建的对象。基本用法如下:

  1. function her(name, sex, height){
  2. this.name = name;
  3. this.sex = sex;
  4. this.height = height;
  5. this.say = function(){
  6. return this.name + ' is a ' + this.sex + ' has ' + this.height;
  7. }
  8. }
  9. var she1 = new her('Anna', 'women', '170cm');
  10. console.log(she1.say()); // 'Anna is a Women has 170cm'

  上面的例子是个非常简单的构造函数模式,但是有点小问题。首先是使用继承很麻烦了,其次output()在每次创建对象的时候都重新定义了,最好的方法是让所有Car类型的实例都共享这个output()方法,这样如果有大批量的实例的话,就会节约很多内存(这种情况在之前的文章里提到过了,可以看一下)。

解决这个问题,我们可以使用如下方式:

  1. function her(name, sex, height){
  2. this.name = name;
  3. this.sex = sex;
  4. this.height = height;
  5. this.say = gaoHan;
  6. }
  7. function gaoHan(){
  8. return this.name + ' is a ' + this.sex + ' has ' + this.height;
  9. }

这个方式虽然可用,但是我们有如下更好的方式。

JavaScript里函数有个原型属性叫prototype,当调用构造函数创建对象的时候,所有该构造函数原型的属性在新创建对象上都可用。按照这样,多个her对象实例可以共享同一个原型,我们再扩展一下上例的代码:

  1. function her(name, sex, height){
  2. this.name = name;
  3. this.sex = sex;
  4. this.height = height;
  5. }
  1. /*注意:这里我们使用了Object.prototype.方法名,而不是Object.prototype主要是用来避免重写定义原型prototype对象*/her.prototype.say = function(){ return this.name + ' is a ' + this.sex + ' has ' + this.height;}
  1. var she1 = new her('Anna', 'women', '170cm');
  2. console.log(she1.say()); // 'Anna is a Women has 170cm'

这里,say()单实例可以在所有Car对象实例里共享使用。另外:我们推荐构造函数以大写字母开头,以便区分普通的函数。

只能用new吗?

  1.  
  1. function her(name, sex, height){
  2. this.name = name;
  3. this.sex = sex;
  4. this.height = height;
  5. this.say = function(){
  6. return this.name + ' is a ' + this.sex + ' has ' + this.height;
  7. }
  8. }
  1. //方法1:作为函数调用 her('Anna', 'women', '170cm');
  1. //添加到window对象上 console.log(window.say());
  2.  
  3. //方法2:在另外一个对象的作用域内调用 var o = new Object(); her.call(o, "Anna", "women", "170cm"); console.log(o.say());

该代码的方法1有点特殊,如果不适用new直接调用函数的话,this指向的是全局对象window,我们来验证一下:

  1. // 作为函数调用
  2. var she = her('Anna', 'women', '170cm');
  3. console.log(typeof tom); // 'undefined'
  4. console.log(window.say());

这时候对象tom是undefined,而window.output()会正确输出结果,而如果使用new关键字则没有这个问题,验证如下:

  1. // 作为函数调用
  2. var she = her('Anna', 'women', '170cm');
  3. console.log(typeof tom); // 'undefined'
  4. console.log(tom.say());

强制使用new!

上述的例子展示了不使用new的问题,那么我们有没有办法让构造函数强制使用new关键字呢,答案是肯定的,上代码:

  1. function Her(name, sex, height){
  2. if( !(this instanceof her) ){
  3. return new Car(name, sexr, height);
  4. }
  5. this.name = name;
  6. this.year = year;
  7. this.height= height;
  8. this.say = function(){
  9. return this.name + ' is a ' + women + ' has ' + this.height;
  10. }
  11. }var she1 = new her('Anna', 'women', '170cm');var she2 = her('Lous', 'women', '165cm');console.log(typeof she1); // object;console.log(typeof she2); // object;

通过判断this的instanceof是不是Car来决定返回new Car还是继续执行代码,如果使用的是new关键字,则(this instanceof Car)为真,会继续执行下面的参数赋值,如果没有用new,(this instanceof Car)就为假,就会重新new一个实例返回。

原始包装函数:

  1. // 使用原始包装函数
  2. var s = new String("my javascript");
  3. var n = new Number(520);
  4. var b = new Boolean(true);
  5.  
  6. // 推荐这种
  7. var s = "my javascript";
  8. var n = 520;
  9. var b = true;

推荐,只有在想保留数值状态的时候使用这些包装函数,关于区别可以参考下面的代码:

  1. // 原始string
  2. var greet = "Hello there";
  3. // 使用split()方法分割
  4. greet.split(' ')[0]; // "Hello"
  5. // 给原始类型添加新属性不会报错
  6. greet.smile = true;
  7. // 单没法获取这个值(18章ECMAScript实现里我们讲了为什么)
  8. console.log(typeof greet.smile); // "undefined"
  9.  
  10. // 原始string
  11. var greet = new String("Hello there");
  12. // 使用split()方法分割
  13. greet.split(' ')[0]; // "Hello"
  14. // 给包装函数类型添加新属性不会报错
  15. greet.smile = true;
  16. // 可以正常访问新属性
  17. console.log(typeof greet.smile); // "boolean"

本章主要讲解了构造函数模式的使用方法、调用方法以及new关键字的区别,希望大家在使用的时候有所注意。

javascript --- 设计模式之构造函数模式的更多相关文章

  1. JavaScript设计模式之构造函数模式

    一.构造函数模式概念 构造函数用于创建特定类型的对象——不仅声明了使用过的对象,构造函数还可以接受参数以便第一次创建对象的时候设置对象的成员值.你可以自定义自己的构造函数,然后在里面声明自定义类型对象 ...

  2. [JS设计模式]:构造函数模式(2)

    基本用法 function Car(model, year, miles) { this.model = model; this.year = year; this.miles = miles; th ...

  3. JavaScript设计模式之----组合模式

    javascript设计模式之组合模式 介绍 组合模式是一种专门为创建Web上的动态用户界面而量身制定的模式.使用这种模式可以用一条命令在多个对象上激发复杂的或递归的行为.这可以简化粘合性代码,使其更 ...

  4. 从ES6重新认识JavaScript设计模式(三): 建造者模式

    1 什么是建造者模式? 建造者模式(Builder)是将一个复杂对象的构建层与其表示层相互分离,同样的构建过程可采用不同的表示. 建造者模式的特点是分步构建一个复杂的对象,可以用不同组合或顺序建造出不 ...

  5. 深入理解JavaScript系列(26):设计模式之构造函数模式

    介绍 构造函数大家都很熟悉了,不过如果你是新手,还是有必要来了解一下什么叫构造函数的.构造函数用于创建特定类型的对象——不仅声明了使用的对象,构造函数还可以接受参数以便第一次创建对象的时候设置对象的成 ...

  6. JavaScript设计模式之建造者模式

    一.建造者模式模式概念 建造者模式可以将一个复杂的对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示.也就是说如果我们用了建造者模式,那么用户就需要指定需要建造的类型就可以得到它们,而具体 ...

  7. JavaScript设计模式之工厂模式

    一.工厂模式概念 工厂模式定义一个用于创建对象的接口,这个接口由子类决定实例化哪一个类.该模式使一个类的实例化延迟到了子类.而子类可以重写接口方法以便创建的时候指定自己的对象类型(抽象工厂). 这个模 ...

  8. JavaScript设计模式之代理模式

    一.代理模式概念 代理,顾名思义就是帮助别人做事,GoF对代理模式的定义如下: 代理模式(Proxy),为其他对象提供一种代理以控制对这个对象的访问.代理模式使得代理对象控制具体对象的引用.代理几乎可 ...

  9. JavaScript设计模式之命令模式

    一.命令模式概念 命令模式(Command)的定义是:用来对方法调用进行参数化处理和传送,经过这样处理过的方法调用可以在任何需要的时候执行.也就是说该模式旨在将函数的调用.请求和操作封装成一个单一的对 ...

随机推荐

  1. Android 软键盘弹出时把原来布局顶上去的解决方法

    键盘弹出时,会将布局底部的导航条顶上去. 解决办法: 在mainfest.xml中,在和导航栏相关的activity中加: <activity            android:name=& ...

  2. iOS_UIImage_图片旋转

    一.目的: 有时候我们获得到的图片我们不是我们想要的方向,需要对图片进行旋转.比如:图片旋转90度180度等. 二.实现过程. 1.获取到该UIImage. 2.开启上下文. 3.上下文的具体操作. ...

  3. 【Android】Android应用安装失败及无法打开

    以下是我个人遇到过的APP无法安装的一些问题: 无法安装应用: 手机系统版本过低:不符合应用支持的最低版本.(比如应用只支持Android 4.0以上的手机,而手机是Android2.3的)解决方案: ...

  4. Windows Azure Virtual Machine (31) 迁移Azure虚拟机

    <Windows Azure Platform 系列文章目录> 为什么要写这篇Blog? 之前遇到过很多客户提问: (1)我之前创建的虚拟机,没有加入虚拟网络.现在需要重新加入虚拟机网络, ...

  5. 在jfinal中使用druid,并配置查看权限

    首先导入druid包,然后配置configPlugin @Override public void configPlugin(Plugins me) { /**配置druid数据连接池插件**/ Dr ...

  6. HTML标签理解

    基础回顾 1:<!DOCTYPE HTML> 这个我们都不陌生,它是文档说明,在html页面的第一行就写的是这个.在 html5中只有一个,大小写不敏感.是便于浏览器识别文档类型. 2:& ...

  7. C#进行Visio二次开发之文件导出及另存Web页面

    在我前面很多关于Visio的开发过程中,介绍了各种Visio的C#开发应用场景,包括对Visio的文档.模具文档.形状.属性数据.各种事件等相关的基础处理,以及Visio本身的整体项目应用,虽然时间过 ...

  8. 在WCF数据访问中使用缓存提高Winform字段中文显示速度

    在我们开发基于WCF访问方式的Winform程序的时候,一般情况下需要对界面显示的字段进行中文显示的解析.如果是硬编码进行中文显示,那么除了不方便调整及代码臃肿外,性能上没有什么问题,但是不建议这样处 ...

  9. DIP依赖倒置原则

    一.定义 1.高层模块不应该依赖低层模块,二者都应该依赖抽象 2.抽象不应该依赖于细节.细节应该依赖于抽象 二.层次化 1.简单介绍 结构良好的面向对象架构都具有清晰的层次定义,每个层次通过一个定义良 ...

  10. httpclient 调用WebAPI

    1.创建webapi项目,提供接口方法如下: /// <summary> /// 获取租户.位置下的所有传感器 /// </summary> /// <returns&g ...