作用域
JS中有兩種作用域:全域作用域|局部作用域
#栗子1
console.log(name); //undefined var name = '波妞'; var like = '宗介' console.log(name); //波妞 function fun(){ console.log(name); //波妞 console.log(eat) //ReferenceError: eat is not defined (function(){ console.log(like) //宗介 var eat = '肉' })() } fun();
【相關課程推薦:JavaScript影片教學】
1. name定義在全域,在全域可以存取到,所以(2)列印能夠正確列印;
2. 在函數fun中,如果沒有定義name屬性,那麼會到它的父作用域去找,所以(3) 也能正確列印。
3. 內部環境可以透過作用域鏈存取所有外部環境,但外部環境無法存取內部環境的任何變數和函數。類似單向透明,這就是作用域鏈,所以 (4) 不行而 (5) 可以。
那麼問題來了,為什麼第一個列印是"undefined",而不是"ReferenceError: name is not defined"。原理簡單的說就是JS的變數提升
變數提升:JS在解析程式碼時,會將所有的宣告提前到所在作用域的最前面
栗子2
console.log(name); //undefined var name = '波妞'; console.log(name); //波妞 function fun(){ console.log(name) //undefined console.log(like) //undefined var name = '大西瓜'; var like = '宗介' } fun();
#相當於
var name; console.log(name); //undefined name = '波妞'; console.log(name); //波妞 function fun(){ var name; var like; console.log(name) //undefined console.log(like) //undefined name = '大西瓜'; like = '宗介' console.log(name) //大西瓜 console.log(like) //宗介 } fun();
注意:是提前到目前作用域的最前面
栗子3
printName(); //printName is not a function var printName = function(){ console.log('波妞') } printName(); //波妞
#相當於
var printName; printName(); //printName is not a function printName = function(){ console.log('波妞') } printName(); //波妞
這樣就好理解了,函數表達式在宣告的時候還只是個變數
栗子4
{ var name = '波妞'; } console.log(name) //波妞 (function(){ var name = '波妞'; })() console.log(name) //ReferenceError: name is not defined { let name = '波妞'; } console.log(name) //ReferenceError: name is not defined
從上面的栗子可以看出,不可以草率的認為JS中var聲明的變數的作用範圍就是大括號的起止範圍,ES5並沒有區塊級作用域,實質是函數作用域;ES6中有了let、const定義後,才有了區塊級作用域。
栗子5
function p1() { console.log(1); } function p2() { console.log(2); } (function () { if (false) { function p1() { console.log(3); } }else{ function p2(){ console.log(4) } } p2(); p1() })(); //4 //TypeError: print is not a function
這是一個非常經典的栗子,宣告提前了,但是因為判斷條件為否,所以沒有執行函數體。所以會出現"TypeError: print is not a function"。 while,switch,for同理
閉包
函數與對其狀態即詞法環境(lexical environment)的引用共同構成閉包(closure )。也就是說,閉包可以讓你從內部函數存取外部函數作用域。在JavaScript中,函數在每次建立時產生閉包。
上面的定義來自MDN,簡單講,閉包就是指有權存取另一個函數作用域中變數的函數。
● 閉包的關鍵在於:外部函數呼叫之後其變數物件本來應該被銷毀,但閉包的存在使我們仍然可以存取外部函數的變數物件. ,
//举个例子 function makeFunc() { var name = "波妞"; function displayName() { console.log(name); } return displayName; } var myFunc = makeFunc(); myFunc();
JavaScript中的函數會形成閉包。閉包是由函數以及創建該函數的詞法環境組合而成。這個環境包含了這個閉包建立時所能存取的所有局部變數
在範例中,myFunc 是執行makeFunc 時所建立的displayName 函式實例的引用,而displayName 實例仍可存取其詞法作用域中的變量,即可以存取name 。由此,當 myFunc 被呼叫時,name 仍可被訪問,其值 '波妞' 就被傳遞到console.log中。 建立閉包最常見方式,就是在一個函數內部建立另一個函數
#● 通常,函數的作用域及其所有變數都會在函數執行結束後被銷毀。但是,在創建了一個閉包以後,這個函數的作用域就會一直保存到閉包不存在為止
//例二 function makeAdder(x) { return function(y) { return x + y; }; } var add5 = makeAdder(5); var add10 = makeAdder(10); console.log(add5(2)); // 7 console.log(add10(2)); // 12 //释放对闭包的引用 add5 = null; add10 = null;
從本質上講,makeAdder 是一個函數工廠— 他創建了將指定的值和它的參數相加求和的函數。在上面的範例中,我們使用函數工廠建立了兩個新函數 — 一個將其參數和 5 求和,另一個和 10 求和。
add5 和 add10 都是閉包。它們共享相同的函數定義,但是保存了不同的詞法環境。在 add5 的環境中,x 為 5。而在 add10 中,x 則為 10。
閉包的作用域鏈包含它自己的作用域,以及包含它的函數的作用域和全域作用域。
● 閉套件只能取得包含函數中的任何變數的最後一個值
//栗子1 function arrFun1(){ var arr = []; for(var i = 0 ; i < 10 ; i++){ arr[i] = function(){ return i } } return arr } console.log(arrFun1()[9]()); //10 console.log(arrFun1()[1]()); //10 //栗子2 function arrFun2(){ var arr = []; for(var i = 0 ; i < 10 ; i++){ arr[i] = function(num){ return function(){ return num }; }(i) } return arr } console.log(arrFun2()[9]()); //9 console.log(arrFun2()[1]()); //1
栗子1 中,arr陣列中包含10個匿名函數,每個函數都可以存取外部的變數i , arrFun1 執行後,其作用域被銷毀,但它的變數依然存在記憶體中,能被循環中的匿名函數訪問,這是的i 為10;
栗子2 中,arr數組中有是個匿名函數,其匿名函數內還有匿名函數,最內層匿名函數訪問的num 被上一級匿名函數保存在了內存中,所以可以訪問到每次的i 的值。
本文來自 js教學 欄目,歡迎學習!
以上是js作用域和閉包詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

JavaScript在Web開發中的主要用途包括客戶端交互、表單驗證和異步通信。 1)通過DOM操作實現動態內容更新和用戶交互;2)在用戶提交數據前進行客戶端驗證,提高用戶體驗;3)通過AJAX技術實現與服務器的無刷新通信。

理解JavaScript引擎內部工作原理對開發者重要,因為它能幫助編寫更高效的代碼並理解性能瓶頸和優化策略。 1)引擎的工作流程包括解析、編譯和執行三個階段;2)執行過程中,引擎會進行動態優化,如內聯緩存和隱藏類;3)最佳實踐包括避免全局變量、優化循環、使用const和let,以及避免過度使用閉包。

Python更適合初學者,學習曲線平緩,語法簡潔;JavaScript適合前端開發,學習曲線較陡,語法靈活。 1.Python語法直觀,適用於數據科學和後端開發。 2.JavaScript靈活,廣泛用於前端和服務器端編程。

Python和JavaScript在社區、庫和資源方面的對比各有優劣。 1)Python社區友好,適合初學者,但前端開發資源不如JavaScript豐富。 2)Python在數據科學和機器學習庫方面強大,JavaScript則在前端開發庫和框架上更勝一籌。 3)兩者的學習資源都豐富,但Python適合從官方文檔開始,JavaScript則以MDNWebDocs為佳。選擇應基於項目需求和個人興趣。

從C/C 轉向JavaScript需要適應動態類型、垃圾回收和異步編程等特點。 1)C/C 是靜態類型語言,需手動管理內存,而JavaScript是動態類型,垃圾回收自動處理。 2)C/C 需編譯成機器碼,JavaScript則為解釋型語言。 3)JavaScript引入閉包、原型鍊和Promise等概念,增強了靈活性和異步編程能力。

不同JavaScript引擎在解析和執行JavaScript代碼時,效果會有所不同,因為每個引擎的實現原理和優化策略各有差異。 1.詞法分析:將源碼轉換為詞法單元。 2.語法分析:生成抽象語法樹。 3.優化和編譯:通過JIT編譯器生成機器碼。 4.執行:運行機器碼。 V8引擎通過即時編譯和隱藏類優化,SpiderMonkey使用類型推斷系統,導致在相同代碼上的性能表現不同。

JavaScript在現實世界中的應用包括服務器端編程、移動應用開發和物聯網控制:1.通過Node.js實現服務器端編程,適用於高並發請求處理。 2.通過ReactNative進行移動應用開發,支持跨平台部署。 3.通過Johnny-Five庫用於物聯網設備控制,適用於硬件交互。

我使用您的日常技術工具構建了功能性的多租戶SaaS應用程序(一個Edtech應用程序),您可以做同樣的事情。 首先,什麼是多租戶SaaS應用程序? 多租戶SaaS應用程序可讓您從唱歌中為多個客戶提供服務


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

MinGW - Minimalist GNU for Windows
這個專案正在遷移到osdn.net/projects/mingw的過程中,你可以繼續在那裡關注我們。 MinGW:GNU編譯器集合(GCC)的本機Windows移植版本,可自由分發的導入函式庫和用於建置本機Windows應用程式的頭檔;包括對MSVC執行時間的擴展,以支援C99功能。 MinGW的所有軟體都可以在64位元Windows平台上運作。

SublimeText3 英文版
推薦:為Win版本,支援程式碼提示!

SublimeText3漢化版
中文版,非常好用

SAP NetWeaver Server Adapter for Eclipse
將Eclipse與SAP NetWeaver應用伺服器整合。

PhpStorm Mac 版本
最新(2018.2.1 )專業的PHP整合開發工具