深入理解new運算符

来源:https://www.cnblogs.com/jofun/archive/2020/04/19/12731473.html
-Advertisement-
Play Games

在 JavaScript 中,new 運算符創建一個用戶定義的對象類型的實例或具有構造函數的內置對象的實例。創建一個對象很簡單,為什麼我們還要多此一舉使用 new 運算符呢?它到底有什麼樣的魔力? ...


在 JavaScript 中,new 運算符創建一個用戶定義的對象類型的實例或具有構造函數的內置對象的實例。創建一個對象很簡單,為什麼我們還要多此一舉使用 new 運算符呢?它到底有什麼樣的魔力?

認識 new 運算符

通過下麵的例子理解 new 運算符:

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

Person.prototype.getName = function () {
  console.log(this.name)
}

var joe = new Person('joe')

joe.sayHello = function () {
  console.log('Hello!')
}

joe.getName() // joe
joe.sayHello() // Hello!

Person.sayHello() // Uncaught TypeError: Person.sayHello is not a function

Person 是一個普通的函數,當它與 new 運算符一起使用時,Person 就是一個構造函數。通過 new Person('joe') 得到的新對象 joe 繼承了 Person 的屬性,同時,this 也指向 joe 實例。為 joe 添加的屬性 sayHello 不會影響 Person,即 joe 是區別與 Person 的一個新對象。

因此,通過 new 創建的實例對象和構造函數之間建立了一條原型鏈,並通過原型鏈賦予實例對象繼承屬性的能力

new 的原理和實現

通過上面的分析,new 運算符內部做瞭如下四個操作:

  • 創建一個空的簡單 JavaScript 對象(即{});
  • 鏈接新對象(即設置該新對象的構造函數)到函數對象;
  • 將新創建的對象作為 this 的上下文;
  • 如果該函數沒有返回對象,返回新創建的對象。

new 的實現如下:

function newOperator (ctor, ...args) {
  var obj = {};
  obj.__proto__ = ctor.prototype
  var res = ctor.apply(obj, args)
  return res || obj;
}

優化一下代碼:

function newOperator (ctor, ...args) {
  var o = Object.create(ctor.prototype) // 合併第一和第二步:創建一個空的簡單 JavaScript 對象(即{}),鏈接新對象(即設置該新對象的構造函數)到函數對象
  return fn.apply(o, args) || o
}

使用 newOperator 函數測試上面 Person 的例子:

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

Person.prototype.getName = function () {
  console.log(this.name)
}

var joe = newOperator(Person, 'joe')

joe.sayHello = function () {
  console.log('Hello!')
}

joe.getName() // joe
joe.sayHello() // Hello!

Person.sayHello() // Uncaught TypeError: Person.sayHello is not a function

結果是一致的。

更好的檢查方式是:

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

console.log(new Person('joe')) // @1
console.log(newOperator(Person, 'joe')) // @2

@1 和 @2 在控制台的顯示信息是一模一樣的。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在日常開發中,我們可能會遇到將一個數組中裡面的重覆值去除,那麼,我就將我自己所學習到的幾種方法分享出來 去除數組重覆值方法: 1,利用indexOf()方法去除 思路:創建一個新數組,然後迴圈要去重的數組,然後用新數組去找要去重數組的值,如果找不到則使用.push添加到新數組,最後把新數組返回回去就 ...
  • 選項卡是前端常見的基本功能,它是用多個標簽頁來區分不同內容,通過選擇標簽快速切換內容。學習本教程之前,讀者需要具備html和css技能,同時需要有簡單的javascript基礎。 先來完成html部分。首先,需要一個元素把整個選項卡包含在內。新建一個div元素,它的id命名為tabBox,如下所示: ...
  • 作用域和閉包-執行上下文: 變數提升(寫代碼時千萬不要先使用再定義) <script> console.log(a);//undefined var a=10; fn('cyy',18); function fn(name,age){ age=20; console.log(name,age);// ...
  • 在前端面試時,面試官經常會問:瀏覽器是多進程還是單進程? 瀏覽器是多進程的,瀏覽器每一個 tab 標簽都代表一個獨立的進程(也不一定,因為多個空白 tab 標簽會合併成一個進程),瀏覽器內核(瀏覽器渲染進程)屬於瀏覽器多進程中的一種。 瀏覽器每個進程有多個線程,主要有以下線程 1)GUI 渲染線程: ...
  • VSCode卸載後進行重新安裝,發現新安裝的還有原來的一些配置,卸載的不徹底,有時候也容易出問題,可按照如下方法卸載乾凈: 1.進入控制面板卸載VSCode,也可以在VSCode的安裝目錄下用程式自帶的卸載程式(這個沒有親自試過.....) 2.這樣卸載完後還有一些配置文件,要想完全卸載,還需要將一 ...
  • 一、承接連載4 3.null和undefined沒有toString()方法,調用就會報錯 var num1 = undefined; console.log(num1.toString()); ​ var num2 = null; console.log(num2.toString()); 二、S ...
  • 定時器功能分析 核心思路: 定義一個變數,根據定時器,每秒執行一次,每次執行++自增操作 變數存儲的數值,就會每秒+1 現在需要的記錄效果,是每0.01秒,也就是10毫秒執行一次 根據累計的數值,執行進位 ms 如果達到 100,就是1秒 如果 秒 達到 60 就是 1分鐘 如果 分鐘 達到 60  ...
  • JavaScript與函數式編程 絕大多數編程語言都會有函數的概念(或者說所有的?我不太確定),他們都可以做出類似的操作: 但是Javascript更適合函數式編程,因為函數對於js來說,是 一等公民 。 我們可以把匿名函數賦值給一個變數,比如: 然後我們可以將這個函數賦值給另一個變數: 這樣做和直 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...