在线观看不卡亚洲电影_亚洲妓女99综合网_91青青青亚洲娱乐在线观看_日韩无码高清综合久久

鍍金池/ 教程/ HTML/ 設(shè)計模式之構(gòu)造函數(shù)模式
代碼復(fù)用模式(避免篇)
S.O.L.I.D 五大原則之接口隔離原則 ISP
設(shè)計模式之狀態(tài)模式
JavaScript 核心(晉級高手必讀篇)
設(shè)計模式之建造者模式
JavaScript 與 DOM(上)——也適用于新手
設(shè)計模式之中介者模式
設(shè)計模式之裝飾者模式
設(shè)計模式之模板方法
設(shè)計模式之外觀模式
強大的原型和原型鏈
設(shè)計模式之構(gòu)造函數(shù)模式
揭秘命名函數(shù)表達式
深入理解J avaScript 系列(結(jié)局篇)
執(zhí)行上下文(Execution Contexts)
函數(shù)(Functions)
《你真懂 JavaScript 嗎?》答案詳解
設(shè)計模式之適配器模式
設(shè)計模式之組合模式
設(shè)計模式之命令模式
S.O.L.I.D 五大原則之單一職責(zé) SRP
編寫高質(zhì)量 JavaScript 代碼的基本要點
求值策略
閉包(Closures)
對象創(chuàng)建模式(上篇)
This? Yes,this!
設(shè)計模式之代理模式
變量對象(Variable Object)
S.O.L.I.D 五大原則之里氏替換原則 LSP
面向?qū)ο缶幊讨话憷碚?/span>
設(shè)計模式之單例模式
Function 模式(上篇)
S.O.L.I.D 五大原則之依賴倒置原則 DIP
設(shè)計模式之迭代器模式
立即調(diào)用的函數(shù)表達式
設(shè)計模式之享元模式
設(shè)計模式之原型模式
根本沒有“JSON 對象”這回事!
JavaScript 與 DOM(下)
面向?qū)ο缶幊讨?ECMAScript 實現(xiàn)
全面解析 Module 模式
對象創(chuàng)建模式(下篇)
設(shè)計模式之職責(zé)鏈模式
S.O.L.I.D 五大原則之開閉原則 OCP
設(shè)計模式之橋接模式
設(shè)計模式之策略模式
設(shè)計模式之觀察者模式
代碼復(fù)用模式(推薦篇)
作用域鏈(Scope Chain)
Function 模式(下篇)
設(shè)計模式之工廠模式

設(shè)計模式之構(gòu)造函數(shù)模式

介紹

構(gòu)造函數(shù)大家都很熟悉了,不過如果你是新手,還是有必要來了解一下什么叫構(gòu)造函數(shù)的。構(gòu)造函數(shù)用于創(chuàng)建特定類型的對象——不僅聲明了使用的對象,構(gòu)造函數(shù)還可以接受參數(shù)以便第一次創(chuàng)建對象的時候設(shè)置對象的成員值。你可以自定義自己的構(gòu)造函數(shù),然后在里面聲明自定義類型對象的屬性或方法。

基本用法

在 JavaScript 里,構(gòu)造函數(shù)通常是認為用來實現(xiàn)實例的,JavaScript 沒有類的概念,但是有特殊的構(gòu)造函數(shù)。通過 new 關(guān)鍵字來調(diào)用定義的否早函數(shù),你可以告訴 JavaScript 你要創(chuàng)建一個新對象并且新對象的成員聲明都是構(gòu)造函數(shù)里定義的。在構(gòu)造函數(shù)內(nèi)部,this 關(guān)鍵字引用的是新創(chuàng)建的對象?;居梅ㄈ缦拢?/p>

function Car(model, year, miles) {
    this.model = model;
    this.year = year;
    this.miles = miles;
    this.output= function () {
        return this.model + "走了" + this.miles + "公里";
    };
}
var tom= new Car("大叔", 2009, 20000);
var dudu= new Car("Dudu", 2010, 5000);
console.log(tom.output());
console.log(dudu.output());

上面的例子是個非常簡單的構(gòu)造函數(shù)模式,但是有點小問題。首先是使用繼承很麻煩了,其次 output()在每次創(chuàng)建對象的時候都重新定義了,最好的方法是讓所有 Car 類型的實例都共享這個 output()方法,這樣如果有大批量的實例的話,就會節(jié)約很多內(nèi)存。

解決這個問題,我們可以使用如下方式:

function Car(model, year, miles) {
    this.model = model;
    this.year = year;
    this.miles = miles;
    this.output= formatCar;
}
function formatCar() {
    return this.model + "走了" + this.miles + "公里";
}

這個方式雖然可用,但是我們有如下更好的方式。

構(gòu)造函數(shù)與原型

JavaScript 里函數(shù)有個原型屬性叫 prototype,當調(diào)用構(gòu)造函數(shù)創(chuàng)建對象的時候,所有該構(gòu)造函數(shù)原型的屬性在新創(chuàng)建對象上都可用。按照這樣,多個 Car 對象實例可以共享同一個原型,我們再擴展一下上例的代碼:

function Car(model, year, miles) {
    this.model = model;
    this.year = year;
    this.miles = miles;
}
/*
注意:這里我們使用了Object.prototype.方法名,而不是Object.prototype
主要是用來避免重寫定義原型prototype對象
*/
Car.prototype.output= function () {
    return this.model + "走了" + this.miles + "公里";
};
var tom = new Car("大叔", 2009, 20000);
var dudu = new Car("Dudu", 2010, 5000);
console.log(tom.output());
console.log(dudu.output());

這里,output()單實例可以在所有 Car 對象實例里共享使用。

另外:我們推薦構(gòu)造函數(shù)以大寫字母開頭,以便區(qū)分普通的函數(shù)。

只能用 new 嗎?

上面的例子對函數(shù) car 都是用 new 來創(chuàng)建對象的,只有這一種方式么?其實還有別的方式,我們列舉兩種:

function Car(model, year, miles) {
    this.model = model;
    this.year = year;
    this.miles = miles;
    // 自定義一個output輸出內(nèi)容
    this.output = function () {
        return this.model + "走了" + this.miles + "公里";
    }
}
//方法1:作為函數(shù)調(diào)用
Car("大叔", 2009, 20000);  //添加到window對象上
console.log(window.output());
//方法2:在另外一個對象的作用域內(nèi)調(diào)用
var o = new Object();
Car.call(o, "Dudu", 2010, 5000);
console.log(o.output()); 

該代碼的方法 1 有點特殊,如果不適用 new 直接調(diào)用函數(shù)的話,this 指向的是全局對象 window,我們來驗證一下:

//作為函數(shù)調(diào)用
var tom = Car("大叔", 2009, 20000);
console.log(typeof tom); // "undefined"
console.log(window.output()); // "大叔走了20000公里"

這時候?qū)ο?tom 是 undefined,而 window.output()會正確輸出結(jié)果,而如果使用 new 關(guān)鍵字則沒有這個問題,驗證如下:

//使用new 關(guān)鍵字
var tom = new Car("大叔", 2009, 20000);
console.log(typeof tom); // "object"
console.log(tom.output()); // "大叔走了20000公里"

強制使用 new

上述的例子展示了不使用 new 的問題,那么我們有沒有辦法讓構(gòu)造函數(shù)強制使用 new 關(guān)鍵字呢,答案是肯定的,上代碼:

function Car(model, year, miles) {
    if (!(this instanceof Car)) {
        return new Car(model, year, miles);
    }
    this.model = model;
    this.year = year;
    this.miles = miles;
    this.output = function () {
        return this.model + "走了" + this.miles + "公里";
    }
}
var tom = new Car("大叔", 2009, 20000);
var dudu = Car("Dudu", 2010, 5000);
console.log(typeof tom); // "object"
console.log(tom.output()); // "大叔走了20000公里"
console.log(typeof dudu); // "object"
console.log(dudu.output()); // "Dudu走了5000公里"

通過判斷 this 的 instanceof 是不是 Car 來決定返回 new Car 還是繼續(xù)執(zhí)行代碼,如果使用的是 new 關(guān)鍵字,則(this instanceof Car)為真,會繼續(xù)執(zhí)行下面的參數(shù)賦值,如果沒有用 new,(this instanceof Car)就為假,就會重新 new 一個實例返回。

原始包裝函數(shù)

JavaScript 里有 3 中原始包裝函數(shù):number,string,boolean,有時候兩種都用:

// 使用原始包裝函數(shù)
var s = new String("my string");
var n = new Number(101);
var b = new Boolean(true);
// 推薦這種
var s = "my string";
var n = 101;
var b = true;

推薦,只有在想保留數(shù)值狀態(tài)的時候使用這些包裝函數(shù),關(guān)于區(qū)別可以參考下面的代碼:

// 原始string
var greet = "Hello there";
// 使用split()方法分割
greet.split(' ')[0]; // "Hello"
// 給原始類型添加新屬性不會報錯
greet.smile = true;
// 單沒法獲取這個值(18章ECMAScript實現(xiàn)里我們講了為什么)
console.log(typeof greet.smile); // "undefined"
// 原始string
var greet = new String("Hello there");
// 使用split()方法分割
greet.split(' ')[0]; // "Hello"
// 給包裝函數(shù)類型添加新屬性不會報錯
greet.smile = true;
// 可以正常訪問新屬性
console.log(typeof greet.smile); // "boolean"

總結(jié)

本章主要講解了構(gòu)造函數(shù)模式的使用方法、調(diào)用方法以及new關(guān)鍵字的區(qū)別,希望大家在使用的時候有所注意。