從一道前端面試題引發的原理性探究

来源:https://www.cnblogs.com/w3cfed/archive/2020/04/05/vue-react-key.html
-Advertisement-
Play Games

作者 | Function | 前端時空 Vue 和 React 中的 key 到底有什麼用? key 是給每一個 vnode 的唯一 id,依靠 key,我們的 diff 操作可以更準確、更快速。 對於簡單列表頁渲染來說 diff 節點也更快,但會產生一些隱藏的副作用,比如可能不會產生過渡效果,或 ...


作者 | Function | 前端時空

Vue 和 React 中的 key 到底有什麼用?

key 是給每一個 vnode 的唯一 id,依靠 key,我們的 diff 操作可以更準確、更快速。 對於簡單列表頁渲染來說 diff 節點也更快,但會產生一些隱藏的副作用,比如可能不會產生過渡效果,或者在某些節點有綁定數據(表單)狀態,會出現狀態錯位。)

diff 演算法的過程中,先會進行新舊節點的首尾交叉對比,當無法匹配的時候會用新節點的 key 與舊節點進行比對,從而找到相應舊節點。

你以為這樣回答,面試官就能放過你。Too young,To simple。下麵是面試官的反問三連擊:

為什麼更準確??

因為帶 key 就不是就地復用了,在 sameNode 函數 a.key === b.key 對比中可以避免就地復用的情況。所以會更加準確,如果不加 key,會導致之前節點的狀態被保留下來,會產生一系列的 bug。

為什麼更快速?

key 的唯一性可以被 Map 數據結構充分利用,相比於遍歷查找的時間複雜度 O(n),Map 的時間複雜度僅僅為 O(1)。

為什麼 Map 數據結構會更快?

Map / Set / WeakSet / WeakMap 就是使用隱藏的 Hash code 優化哈希表

ECMAScript 2015 引入了幾個新的數據結構,例如 Map,Set,WeakSet和 WeakMap,所有這些結構都在後臺使用哈希表。下麵詳細介紹了V8 v6.3+ 如何將 key 存儲在哈希表中的最新進展。

哈希碼 Hash code

散列函數用於將給定的 key 映射到哈希表中的特定位置。一個哈希碼是給定的 key 運行此散列函數的運算結果。
hashCode = hashFunc(key)

在 V8 中,哈希碼只是一個隨機數,與對象值無關。因此,我們無法重新計算它,這意味著我們必須存儲它。

以前,對於那些把 JavaScript 對象作為 key 的情況,V8 將哈希碼作為私有符號(private symbol)存儲在對象上。V8 中的私有符號類似於Symbol,只是它不可枚舉,也不會不會泄漏到用戶空間 JavaScript 中。也就是說這個 symbol 只在 V8 引擎內部使用,用戶的 JavaScript 代碼訪問不到。

function GetObjectHash(key) {
  let hash = key[hashCodeSymbol];
  if (IS_UNDEFINED(hash)) {
    hash = (MathRandom() * 0x40000000) | 0;
    if (hash === 0) hash = 1;
    key[hashCodeSymbol] = hash;
  }
  return hash;
}

之所以行之有效,是因為在將對象添加到哈希表之前,我們不必為哈希碼欄位保留記憶體。當對象被添加到哈希表時,才把新的私有符號存儲在對象上。

與使用內聯緩存(IC)系統進行的任何其他屬性查找一樣,V8 還可以優化哈希碼符號查找,從而為哈希碼提供非常快速的查找。當鍵具有相同的隱藏類時,這對於單態內聯緩存查找非常有效。但是,大多數現實世界的代碼都不遵循這種模式,並且鍵通常具有不同的隱藏類,導致散列碼的復態內聯緩存查找變慢。

私有符號方法的另一個問題是,它在存儲哈希碼時觸發了密鑰的隱藏類轉換。這導致不僅對哈希代碼查找也為上鍵和其他財產查找差的多形態代碼去優化從優化代碼。

JavaScript 對象支持存儲

V8 的 JavaScript 對象(JSObject)使用 2 個 word(除了它的頭部):一個 word 用於存儲指向元素存儲的指針,另一個 word 用於存儲指向屬性存儲的指針。
  • word (computer architecture)

元素存儲用於像數組索引的屬性,而屬性存儲用於其鍵為字元串或符號的屬性。有關這些的更多信息,請參見 Camillo Bruni 的 V8 博客文章。

const x = {};
x[1] = 'bar';      // ← stored in elements
x['foo'] = 'bar';  // ← stored in properties

隱藏哈希碼 Hiding the hash code

存儲哈希碼最簡單的方法是將 JavaScript 對象的大小擴展一個字,並將散列碼直接存儲在對象上。但是,對於那些沒有添加到哈希表中的對象,這會浪費記憶體。相反,我們可以嘗試將散列碼存儲在元素存儲或屬性存儲中。

元素存儲是一個包含其長度和所有元素的數組。在這裡沒有太多的工作要做,因為可以把哈希碼存儲在一個保留的槽中(比如第 0 個索引),不過,當我們不使用這個對象作為哈希表中的關鍵字時,仍然會浪費記憶體。

讓我們看看屬性存儲。有兩種數據結構用作屬性存儲:數組字典

與元素存儲中使用的數組不同,元素存儲不具有上限,而屬性存儲中使用的數組的上限為 1022 個值。由於性能原因,V8 在超過此限制時則轉換為使用字典模式。(我略微簡化了這一點 - V8 也可以在其他情況下使用字典,但是可以存儲在數組中的值的數量有一個固定的上限。)

因此,屬性存儲有三種可能的狀態:

  • 空(沒有屬性)
  • 數組(最多可以存儲 1022 個值)
  • 字典

1、屬性存儲是空的

對於空的情況,我們可以直接在 JSObject 的偏移量上存儲哈希碼。

2、屬性存儲是一個數組

V8 表示小於 231 的整數(在 32 位系統上)更加高效,如 Smi。在一個 Smi 中,最低有效位是用來區別指針的 tag,而其餘的 31 位保存實際的整數值。

通常,數組將它們的長度存儲為 Smi。既然我們知道這個數組的最大容量只有 1022 個,我們只需要 10 個比特就可以存儲這個長度。我們可以使用剩下的 21 位來存儲哈希碼!

3、屬性支持存儲是一個字典

對於字典的情況,我們將字典大小增加1個字,以便將哈希碼存儲在字典起始位置的專用槽中。在這種情況下,我們可能會浪費掉一個字的存儲空間,因為這個比例增長的大小並不像數組那麼大。

通過這些更改,哈希碼查找不再需要經過複雜的 JavaScript 屬性查找機制。

性能改進

SixSpeed 對 Map 和 Set 的基準測試,這些變化導致了 5〜50% 的性能提升。

這一變化也導致 ARES6 中的基準測試提高了 5%。

這也導致 Emberperf 基準測試套件中測試的 Ember.js 提高了 18%。

探究總結

掌握一門技術併合理使用它的最好辦法就是深入理解這項技術背後的工作原理

參考資料

https://v8.dev/blog/hash-code

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

-Advertisement-
Play Games
更多相關文章
  • 先安裝zsh 安裝oh my zsh 手動安裝 自動安裝 ` 不管使用手動 自動安裝,完成後都需要重啟。oh my zsh才能生效。 修改主題 挑選你喜歡的主題:https://github.com/robbyrussell/oh my zsh/wiki/Themes ...
  • Redis是一個開源的、基於記憶體的數據結構存儲器,可以用作資料庫、緩存和消息中間件 Redis最常用的功能 緩存 分散式鎖 ...
  • 創建3台虛擬機 主機為桌面版 其他為迷你版本 ******************************常用命令、進程名稱****************************啟動集群命令: start-all.sh啟動zookeeper: zkServer.sh start 啟動journal ...
  • 統計一個表的數據量是經常遇到的需求,但是不同的表設計及不同的寫法,統計性能差別會有較大的差異,下麵就簡單通過實驗進行測試(大家測試的時候註意緩存的情況,否則影響測試結果)。 1、 準備工作 為了後續測試工作的進行,先準備幾張用於測試的表及數據,為了使測試數據具有參考意義,建議測試表的數據量大一點,以 ...
  • 簡介 Flutter Fly是什麼?Flutter Fly是一款開源的Flutter 項目,非常適合初學者進行學習。App內集成了160+Flutter基礎控制項的詳細介紹及用法,內容來源於: "http://laomengit.com/" 。 歡迎頁: 首頁、控制項頁面、詳情頁及搜索頁面: 我: Ap ...
  • 質數是指在大於1的自然數中,除了1和它自身外沒有其他因數的自然數。 一、標記法,flag初始值為true,當n%i 0時(1<i<n),說明n不是質數,此時flag值為false且迴圈終止;當n%i != 0 時,flag的值始終為true,此時會輸出n是質數。 <!DOCTYPE html> <h ...
  • 1 <!DOCTYPE html> 2 <html lang="en"> 3 <head> 4 <meta charset="UTF-8"> 5 <meta name="viewport" content="width=device-width, initial-scale=1.0"> 6 <tit ...
  • Vue.js作為目前最熱門最具前景的前端框架之一,其提供了一種幫助我們快速構建並開發前端項目的新的思維模式。本文旨在幫助大家認識Vue.js,並詳細介紹使用vue-cli腳手架工具快速的創建Vue項目,以及對項目目錄結構的解釋說明,使大家清晰的瞭解Vue項目的開發流程。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...