JavaScript 的編譯與執行原理

文章目錄

  • 前言
    • 🧠 一、JavaScript 編譯與執行過程
      • 1. 編譯階段(發生在代碼執行前)
        • ? 1.1 詞法分析(Lexical Analysis)
        • ? 1.2 語法分析(Parsing)
        • ? 1.3 語義分析與生成執行上下文
    • 🧰 二、執行階段
      • ? 2.1 創建執行上下文(Execution Context)
      • ? 2.2 變量環境 vs 詞法環境
    • 🔁 三、作用域、作用域鏈與閉包
      • ? 3.1 作用域(Scope)
      • ? 3.2 作用域鏈(Scope Chain)
      • ? 3.3 閉包(Closure)
    • 🔄 四、循環作用域與經典陷阱
      • ? 4.1 for 循環中 var 的作用域陷阱
    • 🔍 五、執行上下文棧(ECS)
    • 🔐 六、總結:如何理解閉包與作用域鏈的結合
    • 📌 圖示記憶(簡化示意)
    • 總結


前言

系統深入地講解 JavaScript 的編譯與執行原理,并結合作用域、閉包、作用域鏈等核心概念,構建一個完整的知識體系。


🧠 一、JavaScript 編譯與執行過程

雖然 JavaScript 是解釋型語言,但它在執行前仍然會經歷編譯階段(由現代 JavaScript 引擎完成,如 V8)。整體過程如下:

1. 編譯階段(發生在代碼執行前)

包括以下幾步:

? 1.1 詞法分析(Lexical Analysis)
  • 將源代碼拆解成詞法單元(Token)
  • 例如:let a = 10;leta=10;
  • 負責這一階段的是詞法分析器(Lexer)。
// 代碼
let x = 10 + y;// 生成的tokens
[{ type: 'Keyword', value: 'let' },{ type: 'Identifier', value: 'x' },{ type: 'Punctuator', value: '=' },{ type: 'Numeric', value: '10' },{ type: 'Punctuator', value: '+' },{ type: 'Identifier', value: 'y' },{ type: 'Punctuator', value: ';' }
]

關聯知識:
詞法環境是 JavaScript 引擎內部用來管理變量和函數作用域的機制,它是理解作用域和閉包的核心概念。

詞法環境的組成
一個詞法環境包含兩部分:

??環境記錄(Environment Record)??:存儲變量和函數聲明的實際位置
??對外部詞法環境的引用(Outer Lexical Environment)??:形成作用域鏈

// 示例代碼
let x = 10;function foo() {let y = 20;console.log(x + y);
}// 對應的詞法環境結構
globalLexicalEnvironment = {environmentRecord: { x: 10, foo: <function> },outer: null
}fooLexicalEnvironment = {environmentRecord: { y: 20 },outer: globalLexicalEnvironment
}

詞法環境的特性
??靜態性??:在代碼編寫階段就已確定(詞法作用域)
??嵌套性??:可以形成多層嵌套結構
??持久性??:被閉包引用的詞法環境不會被銷毀

? 1.2 語法分析(Parsing)
  • 將 Token 轉換為 抽象語法樹(AST)
  • AST 是代碼結構的樹狀表示,每個節點代表代碼結構的一個成分。
  • 語法分析器(Parser)處理這一步。
let x = 10 + y;
// 生成的AST結構
{type: "VariableDeclaration",kind: "let",declarations: [{type: "VariableDeclarator",id: { type: "Identifier", name: "x" },init: {type: "BinaryExpression",operator: "+",left: { type: "Literal", value: 10 },right: { type: "Identifier", name: "y" }}}]
}
? 1.3 語義分析與生成執行上下文
  • 變量提升作用域環境創建函數聲明處理

  • 此時創建:

    • 執行上下文棧(Execution Context Stack)
    • 詞法環境(Lexical Environment)
    • 變量環境(Variable Environment)

🧰 二、執行階段

編譯完成后,進入代碼執行:

? 2.1 創建執行上下文(Execution Context)

每個函數/全局代碼執行時,會創建一個上下文環境:

  • 包含:

    • 變量環境(變量/函數聲明)
    • 詞法環境(作用域)
    • this 綁定
    • 外部環境引用(outer)

? 2.2 變量環境 vs 詞法環境

  • 變量環境

    • 存儲變量、函數聲明(var/函數聲明)
  • 詞法環境(Lexical Environment):

    • 變量環境 + 外部環境引用(outer)
    • 用于作用域鏈的構建。

🔁 三、作用域、作用域鏈與閉包

? 3.1 作用域(Scope)

  • 定義變量和函數的可訪問范圍

  • 分為:

    • 全局作用域
    • 函數作用域
    • 塊級作用域(let/const)

? 3.2 作用域鏈(Scope Chain)

  • 當前執行上下文的詞法環境中包含對上級詞法環境的引用
  • 在查找變量時,從當前作用域出發,逐層向上查找,直到全局作用域。
function outer() {let a = 10;function inner() {console.log(a); // 通過作用域鏈訪問 outer 的 a}inner();
}
outer();

? 3.3 閉包(Closure)

  • 閉包是函數+定義它的詞法環境的組合
  • 當一個函數“脫離”了它定義時的作用域,仍然“記住”當時的變量。
function makeCounter() {let count = 0;return function () {return ++count;}
}
const counter = makeCounter();
counter(); // 1
counter(); // 2
  • count 保存在 makeCounter 的詞法環境中,被返回的函數形成閉包訪問它。

🔄 四、循環作用域與經典陷阱

? 4.1 for 循環中 var 的作用域陷阱

for (var i = 0; i < 3; i++) {setTimeout(() => {console.log(i); // 輸出三個 3}, 0);
}
  • 原因:var 沒有塊級作用域,i 綁定在同一個詞法環境中。
  • 修復方法:使用 let(創建新的詞法環境)
for (let i = 0; i < 3; i++) {setTimeout(() => {console.log(i); // 輸出 0 1 2}, 0);
}

🔍 五、執行上下文棧(ECS)

每當函數調用時,會創建新的執行上下文并壓棧

  1. 創建全局執行上下文 → 入棧
  2. 調用函數 → 創建函數執行上下文 → 入棧
  3. 函數執行完畢 → 執行上下文出棧

🔐 六、總結:如何理解閉包與作用域鏈的結合

  1. JavaScript 中函數定義時就“捕獲”了其父級詞法環境(靜態作用域)。
  2. 執行時,通過作用域鏈查找變量,先本地再向上查找。
  3. 如果內部函數延遲執行(異步或返回),那么閉包可以“保持”對外部變量的訪問。

📌 圖示記憶(簡化示意)

makeCounter() 創建執行上下文
└── 詞法環境 {count = 0,outer = Global}返回的匿名函數 閉包:
└── 詞法環境 {outer = makeCounter.LE}

在這里插入圖片描述

在這里插入圖片描述
在這里插入圖片描述

總結

  • ??詞法環境??是 JavaScript 作用域管理的核心機制,具有靜態性和嵌套性
  • 循環中使用 let 會為每次迭代創建新的詞法環境副本
  • ??詞法分析??和??語法分析??是編譯的前期階段,與運行時詞法環境不同
  • JavaScript 引擎通過這種機制實現塊級作用域和閉包功能
  • 理解這些概念有助于編寫正確的作用域代碼和調試復雜的作用域問題

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/81350.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/81350.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/81350.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

WORD個人簡歷單頁326款模版分享下載

WORD個人簡歷模版下載&#xff1a;WORD個人簡歷模版https://pan.quark.cn/s/7e79a822c490

Android 中 顯示 PDF 文件內容(AndroidPdfViewer 庫)

PDFView 是一個用于在 Android 應用中顯示 PDF 文檔的庫。它提供了豐富的功能和靈活的配置選項&#xff0c;使得開發者能夠輕松地在應用中嵌入 PDF 閱讀器。 一、 添加依賴 在模塊的 build.gradle 文件中添加以下依賴&#xff1a; // pdfimplementation("com.github.bar…

微信小程序學習之搜索框

1、第一步&#xff0c;我們在index.json中引入vant中的搜索框控件&#xff1a; {"usingComponents": {"van-search": "vant/weapp/search/index"} } 2、第二步&#xff0c;直接在index.wxml中添加布局&#xff1a; <view class"index…

OpenCL C++ 常見屬性與函數

核心對象與屬性 對象/屬性描述示例cl::Platform表示OpenCL平臺cl::Platform::get(&platforms)cl::Device表示計算設備cl::Device::getDefault()cl::Context管理設備、內存和命令隊列的上下文cl::Context(contextDevices)cl::CommandQueue命令隊列,用于提交命令cl::Command…

Milvus 視角看重排序模型(Rerankers)

在信息檢索和生成式人工智能領域&#xff0c;重排序器是優化初始搜索結果順序的重要工具。重排序器與傳統的嵌入模型不同&#xff0c;它將查詢和文檔作為輸入&#xff0c;并直接返回相似度得分&#xff0c;而不是嵌入。該得分表示輸入查詢和文檔之間的相關性。 重排序器通常在…

C語言:gcc 如何調用 Win32 打開文件對話框 ?

在 Windows 平臺上使用 gcc 調用原生 Win32 API 實現文件打開對話框是可行的&#xff0c;但需要直接使用 Win32 的 GetOpenFileName 函數&#xff08;位于 commdlg.h 頭文件&#xff0c;依賴 comdlg32.lib 庫&#xff09;。以下是完整實現步驟和代碼示例&#xff1a; 編寫 file…

計算機視覺與深度學習 | Python實現EMD-SSA-VMD-LSTM時間序列預測(完整源碼和數據)

EMD-SSA-VMD-LSTM混合模型 一、環境配置與依賴二、數據生成&#xff08;示例數據&#xff09;三、多級信號分解1. 經驗模態分解&#xff08;EMD&#xff09;2. 奇異譜分析&#xff08;SSA&#xff09;3. 變分模態分解&#xff08;VMD&#xff09; 四、數據預處理1. 歸一化處理2…

vue配置子路由,實現點擊左側菜單,內容區域顯示不同的內容

文章目錄 一、路由鏈路二、實現步驟準備二級路由下的.vue文件配置子路由聲明router-view標簽為菜單項 el-menu-item 設置index屬性&#xff0c;設置點擊后的路由路徑 三、參考資料 一、路由鏈路 二、實現步驟 準備二級路由下的.vue文件 配置子路由 router/index.js import {…

ModuleNotFoundError: No module named ‘SDToolbox‘

(py311) C:>python Python 3.11.11 | packaged by Anaconda, Inc. | (main, Dec 11 2024, 16:34:19) [MSC v.1929 64 bit (AMD64)] on win32 Type “help”, “copyright”, “credits” or “license” for more information. from SDToolbox import PostShock_eq Tracebac…

Hi3516DV500刷寫固件

hi3516DV500刷固件 1、硬件連接 2、軟件準備 3、刷固件步驟 一、硬件連接 特別注意的是&#xff0c;串口的接線順序 通過網線連接好筆記本和開發板后&#xff0c;需要確認一下網口水晶頭是否閃爍&#xff0c;以確認網絡物理是否連通 二、軟件資源準備 固件包準備 打開工具…

正則表達式r前綴使用指南

正則表達式中的 r&#xff1a;解鎖字符串轉義的魔法 正則表達式是處理字符串的強大工具&#xff0c;但它常常伴隨著轉義字符的復雜性。如果你曾因 \n、\t 或 \\ 的使用而困惑&#xff0c;那么這篇文章將為你揭開謎底&#xff0c;解釋為什么 r 是正則表達式中的「神奇武器」。本…

網絡攻防模擬:城市安全 “數字預演”

在當今數字化快速發展的時代&#xff0c;網絡安全和城市安全面臨著前所未有的挑戰。為有效應對這些挑戰&#xff0c;利用先進的技術搭建模擬演練平臺至關重要。圖撲軟件的 HT for Web 技術&#xff0c;為網絡攻防模擬與城市安全演練提供了全面且高效的解決方案。 三維場景搭建&…

AI模型開發全流程筆記

一、訓練數據準備階段 數據采集標準 格式要求&#xff1a;嚴格QA對形式&#xff08;1問1答&#xff09; 數量基準&#xff1a; 基礎量&#xff1a;500組QA對 優化量&#xff1a;800-1000組QA對 內容規范&#xff1a; 聚焦單一業務節點&#xff08;如售后場景&#xff09; …

1688 數據接口調用秘籍:高效獲取商品實時信息的開發指南

在電商行業競爭白熱化的當下&#xff0c;企業想要搶占市場先機&#xff0c;實時掌握商品信息至關重要。作為國內 B2B 電商巨頭&#xff0c;1688 平臺匯聚海量商品資源&#xff0c;通過高效調用其數據接口獲取商品實時信息&#xff0c;能為企業價格策略制定、庫存管理、競品分析…

milvus學習筆記

本文主要由AI生成&#xff0c;請注意自己查看源代碼校驗。 Milvus v2.4 系統架構概覽 Milvus 采用分布式微服務架構&#xff0c;將計算層&#xff08;Proxy、QueryCoord、QueryNode、IndexCoord、DataCoord、DataNode 等&#xff09;與存儲層&#xff08;Pulsar、MinIO/S3、e…

使用教程:8x16模擬開關陣列可級聯XY腳雙向導通自動化接線

以下通過點亮LED進行基本使用流程演示&#xff0c;實際可以連接復雜外設&#xff08;SPI、CAN、ADC等&#xff09; 單模塊使用 RX、TX、5V和GND接到串口模塊&#xff1b;X5接5V&#xff1b;Y2接LED;LED-接GND 串口模塊插上電腦后&#xff0c;LED沒有亮&#xff1b;因為此時模…

HarmonyOS NEXT~鴻蒙應用上架指南:HarmonyOS應用發布全流程解析

HarmonyOS NEXT&#xff5e;鴻蒙應用上架指南&#xff1a;HarmonyOS應用發布全流程解析 引言 隨著華為鴻蒙操作系統(HarmonyOS)生態的快速發展&#xff0c;越來越多的開發者希望將自己的應用上架到鴻蒙應用市場。本文將詳細介紹鴻蒙應用上架的全流程&#xff0c;幫助開發者順…

20250517 我設想一個空間,無限大,空間不與其中物質進行任何作用,甚至這個空間能容納可以伸縮的空間

1.我設想一個空間&#xff0c;無限大&#xff0c;空間不與其中物質進行任何作用&#xff0c;甚至這個空間能容納可以伸縮的空間 您設想的這個空間具有一些有趣的特點&#xff1a; 無限大&#xff1a;空間本身沒有邊界或限制&#xff0c;理論上可以容納無限多的物質或結構。非…

使用 Kaniko來構建鏡像

使用 Kaniko來構建鏡像 Kaniko 是一種專注于容器鏡像構建的開源工具&#xff0c;其核心設計理念與 Docker 存在顯著差異。以下從功能定位、技術實現和適用場景三方面進行對比分析&#xff1a; 一、Kaniko 的核心特性 無需 Docker 守護進程 Kaniko 直接在容器或 Kubernetes 集…

webman用nginx代理靜態json文件的異步跨域

場景 有.json文件置于webman的public目錄下&#xff0c;使用了nginx做代理&#xff0c;直接訪問文件是可以正常加載的&#xff0c;但跨域瀏覽器就無法加載文件。 nginx配置 文件是否存在于跟目錄&#xff0c;存在則設置請求頭&#xff0c;不存在則將請求交給webman處理即可。…