JS设计模式深入理解—工厂模式、寄生构造函数模式和稳妥构造函数模式的区别

在学习《JavaScript高级程序设计》(第3版)第六章创建对象时,遇到了针对创建自定义类型对象的几种设计模式。其中的工厂模式寄生构造函数模式以及稳妥构造函数模式三者在实现上十分相似,但却具有微妙的差别,所以对它们做一个总结。

一、工厂模式

function createPerson(name, age, job) {
    var o = new Object();
    o.name = name;
    o.age = age;
    o.job = job;
    o.sayName = function() {
        alert(this.name);
    }
    return o;
}

var person1 = createPerson("Nicholas", 29, "Software Engineer");
var person2 = createPerson("Greg", 27, "Doctor");

工厂模式顾名思义,就是通过定义一个通用的函数,将对象的所有创建工作都封装到这个函数中。之后每当需要创建一个对象时,只需要调用这个函数,同时给出初始化对象所需的各个参数,就能自动返回创建好的对象。这就如同工厂里批量生产一件件产品一般,因为创建出的所有对象之间虽然内容不同,但都出自同一模板

二、寄生构造函数模式

function Person(name, age, job) {
    var o = new Object();
    o.name = name;
    o.age = age;
    o.job = job;
    o.sayName = function() {
        alert(this.name);
    }
    return o;
}

var person1 = new Person("Nicholas", 29, "Software Engineer");
var person2 = new Person("Greg", 27, "Doctor");

寄生构造函数模式与工厂模式极为相似,区别在于:

  • 寄生构造函数模式将工厂模式中的那个通用函数createPerson()改名为Person(),并将其看作为对象的构造函数。
  • 创建对象实例时,寄生构造函数模式采用new操作符

那么两者有什么功能上的差别呢?事实上,两者本质上的差别仅在于new操作符(因为函数取什么名字无关紧要),工厂模式创建对象时将createPerson看作是普通的函数,而寄生构造函数模式创建对象时将Person看作是构造函数,不过这对于创建出的对象来说,没有任何差别

对于两者的差别,作者在书中是这么说的:

除了使用new操作符并把使用的包装函数叫做构造函数之外,这个模式跟工厂模式其实是一模一样的。构造函数在不返回值的情况下,默认会返回新对象实例。而通过在构造函数的末尾添加一个return语句,可以重写调用构造函数时返回的值。

根据作者的意思,构造函数和普通函数的区别在于:当使用new+构造函数创建对象时,如果构造函数内部没有return语句,那么默认情况下构造函数将返回一个该类型的实例(如果以上面的例子为参考,person1和person2为Person类型的对象实例,可以使用person1 instanceof Person检验),但如果构造函数内部通过return语句返回了一个其它类型的对象实例,那么这种默认的设置将被打破,构造函数最终返回的实例类型将以return语句中对象实例的类型为准。

基于这个规则,在Person()构造函数中,由于最后通过return语句返回了一个Object类型的对象实例,所以通过该构造函数创建的对象实际上是Object类型而不是Person类型;这样一来就与createPerson()函数返回的对象类型相同,因此可以说工厂模式和寄生构造函数模式在功能上是等价的

如果非要说两者的不同,并且要从其中选择一个作为创建对象的方法的话,我个人更偏向于寄生构造函数模式一些。这是因为new Person()(寄生构造函数模式)更能让我感觉到自己正在创建一个对象,而不是在调用一个函数(工厂模式)。

三、稳妥构造函数模式

function Person(para_name, para_age, para_job) {
    //创建要返回的对象
    var o = {};

    //在这里定义私有属性和方法
    var name = para_name;
    var age = para_age;
    var job = para_job;

    var sayAge = function() {
        alert(age);
    };

    //在这里定义公共方法
    o.sayName = function() {
        alert(name);
    };

    //返回对象
    return o;
}

var person1 = Person("Nicholas", 29, "Software Engineer");    //创建对象实例
person1.sayName();    //Nicholas
person1.name;         //undefined
person1.sayAge();     // 报错

稳妥构造函数模式与前面介绍的两种设计模式具有相似的地方,都是在函数内部定义好对象之后返回该对象来创建实例。然而稳妥构造函数模式的独特之处在于具有以下特点:

  • 没有通过对象定义公共属性
  • 在公共方法中不使用this引用对象自身
  • 不使用new操作符调用构造函数

这种设计模式最适合在一些安全的环境中使用(这些环境中会禁止使用this和new);为了较好地理解这种设计模式,我们可以采取类比的方法——这种构造对象的方式就如同C++/Java语言中通过访问控制符private定义出包含私有成员的类的方式一样(将上例按C++中类的方式来定义):

class Person {
//定义私有成员变量和函数
private: 
    string name;
    int age;
    string job;
    int sayAge() {return age;}
//定义构造函数和公共方法(函数)
public:
    string sayName() {return name;}    //公共方法
    Person(string p_name, int p_age, string p_job):name(p_name),age(p_age),job(p_job) {}  //构造函数
}

//创建对象实例
Person person1("Nicholas", 29, "Software Engineer");
person1.sayName();    //Nicholas
person1.name;         //报错(无法访问)
person1.sayAge();     //报错(无法访问)

可见,利用C++定义出了一个Person类,其中的nameagejob以及sayAge()是私有成员,无法通过类似person1.name的方式直接访问,这是一种类的保护机制;而定义为publicsayName()函数则可以直接访问。

JS中的稳妥构造函数模式正是为了实现这样的数据保护机制。它巧妙地利用了函数的作用域实现了对象属性的私有化:在函数中定义的变量是局部变量,按道理本应该在函数执行完毕退出后进行销毁或清理,但由于通过对象的公共方法对该局部变量保持着引用,所以该变量即便是在构造函数退出之后也依然保持有效(闭包)。

这样一来,创建出的对象既能通过公共方法提供的访问接口对私有属性进行访问(引用的是构造函数的局部变量),也能保证无法通过对象自身对其直接访问(person1.name无法访问到对应数据,因为name是构造函数的局部变量而不是对象的属性),从而保证了对象属性的访问安全。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容