溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

JavaScript模擬實(shí)現(xiàn)new關(guān)鍵字的方法

發(fā)布時(shí)間:2020-08-29 14:55:31 來源:億速云 閱讀:330 作者:小新 欄目:web開發(fā)

這篇文章給大家分享的是有關(guān)JavaScript模擬實(shí)現(xiàn)new關(guān)鍵字的方法的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考。一起跟隨小編過來看看吧。

最近工作太忙,快接近兩周沒更新博客,總感覺有一些事情等著自己去做,雖然工作內(nèi)容對自己提升挺大,但我總覺得,一直埋著頭走路,偶爾也需要抬起頭來,看看現(xiàn)在和自己的期望向是否脫軌,所以周末還是選擇來星巴克寫些文字。

今天記錄 JavaScript 中 new 關(guān)鍵字的模擬實(shí)現(xiàn),當(dāng)我們在模擬實(shí)現(xiàn)某個(gè)語言行為之前,應(yīng)該想想這個(gè)行為都做了哪些事情,通過實(shí)踐,最后也能更加掌握知識(shí)點(diǎn),這就是很多面試題都會(huì)問到模擬實(shí)現(xiàn)的原因,目的是為了考察候選人知識(shí)的深度。

function Person(name) {
    this.name = name;
}
var person = new Person('jayChou');
typeof(person)  // "object"
person instanceof Person  // true
person.__proto__ === Person.prototype  // true
person.constructor === Person  //  true
person.constructor === Person.prototype.constructor  // true

以上,可以看出:

  1. new 創(chuàng)建并返回了一個(gè)新對象,是構(gòu)造函數(shù)的實(shí)例

  2. 對象的實(shí)例的構(gòu)造函數(shù)屬性其實(shí)是構(gòu)造函數(shù)的原型對象的 constructor 屬性

  3. 對象實(shí)例的 __proto__ 關(guān)聯(lián)到構(gòu)造函數(shù)的原型對象

上面的內(nèi)容有關(guān)于 JavaScript 中原型對象和原型鏈的知識(shí),不夠清楚的同學(xué)可以查看我之前的博客。

由于 new 是 JS 的一個(gè)關(guān)鍵字,我們無法實(shí)現(xiàn)關(guān)鍵字,但我們可以通過函數(shù)的形式來模擬 new 關(guān)鍵字的行為。

一、基本思路

知道 new 關(guān)鍵字做了哪些工作,那我們就有了模擬實(shí)現(xiàn)的基本思路。

/**
 * 模擬實(shí)現(xiàn) JavaScript new 操作符
 * @param  {Function} constructor [構(gòu)造函數(shù)]
 * @return {Object|Function|Regex|Date|Error}      [返回結(jié)果]
 */
function mockNew() {
    // 創(chuàng)建一個(gè)空對象
    let resultObj = new Object();

    // 取傳入的第一個(gè)參數(shù),即構(gòu)造函數(shù),并刪除第一個(gè)參數(shù)。
    let constructor =  Array.prototype.shift.call(arguments);
    
    // 類型判斷,錯(cuò)誤處理
    if(typeof constructor !== "function") {
        throw("構(gòu)造函數(shù)第一個(gè)參數(shù)應(yīng)為函數(shù)");
    }
    
    // 綁定 constructor 屬性
    resultObj.constructor = constructor;
    
    // 關(guān)聯(lián) __proto__ 到 constructor.prototype
    resultObj.__proto__ = constructor.prototype;
    
    // 將構(gòu)造函數(shù)的 this 指向返回的對象
    constructor.apply(resultObj, arguments);
    
    // 返回對象
    return resultObj;
}

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


var person = mockNew(Person, "jayChou");

console.log(person);

// constructor: ? Person(name)
// name: "jayChou"
// __proto__: Object

基本思路正確! 所以我們完成了 new 關(guān)鍵字的初步模擬?;锇閭兛梢宰约簞?dòng)手敲一下,每句代碼自己是否都能理解。

二、處理返回值

構(gòu)造函數(shù)也是函數(shù),有不同類型返回值。有時(shí)候構(gòu)造函數(shù)會(huì)返回指定的對象內(nèi)容,所以要對這部分進(jìn)行處理。

/**
 * 模擬實(shí)現(xiàn) JavaScript new 操作符
 * @param  {Function} constructor [構(gòu)造函數(shù)]
 * @return {Object|Function|Regex|Date|Error}      [返回結(jié)果]
 */
function mockNew() {
    // 創(chuàng)建一個(gè)空對象
    let emptyObj = new Object();

    // 取傳入的第一個(gè)參數(shù),即構(gòu)造函數(shù),并刪除第一個(gè)參數(shù)。
    // 關(guān)于為什么要用 Array.prototype.shift.call 的形式,見之前的博客文章 《JavaScript之a(chǎn)rguments》
    let constructor =  Array.prototype.shift.call(arguments);
    
    // 類型判斷,錯(cuò)誤處理
    if(typeof constructor !== "function") {
        throw("構(gòu)造函數(shù)第一個(gè)參數(shù)應(yīng)為函數(shù)");
    }
    
    // 綁定 constructor 屬性
    emptyObj.constructor = constructor;
    
    // 關(guān)聯(lián) __proto__ 到 constructor.prototype
    emptyObj.__proto__ = constructor.prototype;
    
    // 將構(gòu)造函數(shù)的 this 指向返回的對象
    let resultObj = constructor.apply(emptyObj, arguments);
    
    // 返回類型判斷, 如果是對象,則返回構(gòu)造函數(shù)返回的對象
    if (typeof resultObj === "object") {
        return resultObj
    }
    
    // 返回對象
    return emptyObj;
}

function Person(name) {
    this.name = name;
    return {
        name: this.name,
        age: 40
    }
}


var person = mockNew(Person, "jayChou");

console.log(person);

// {name: "jayChou", age: 40}
// age: 40
// name: "jayChou"
// __proto__: Object

當(dāng)返回值返回了一個(gè)自定義對象后,模擬 new 函數(shù)就返回該自定義對象。

JavaScript new 關(guān)鍵字的意義在于讓普通函數(shù)生成一個(gè)新對象,并將對象實(shí)例的 __proto__ 關(guān)聯(lián)到函數(shù)的 prototype 對象。

感謝各位的閱讀!關(guān)于JavaScript模擬實(shí)現(xiàn)new關(guān)鍵字的方法就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI