9 JavaScript閉包

来源:https://www.cnblogs.com/zczhaod/archive/2023/08/21/17645458.html
-Advertisement-
Play Games

##### 9 閉包 我們都知道,函數里是可以訪問函數外的全局變數,而函數外不能訪問函數內的局部變數,如下: ```js // 函數外定義a,在函數內可以訪問 var a = "測試"; function fn() { console.log(a); } fn(); ``` 執行結果: ![imag ...


9 閉包

我們都知道,函數里是可以訪問函數外的全局變數,而函數外不能訪問函數內的局部變數,如下:

// 函數外定義a,在函數內可以訪問
var a = "測試";
function fn() {
    console.log(a);
}
fn();

執行結果:
image

// 函數內定義a,在函數外訪問不到
function fn() {
    var a = "測試";
}
console.log(a);

執行結果:
image

之所以出現這樣的情況,就是因為函數內定義的變數為局部變數,函數外定義的變數為全局變數(不同於windows)。在函數內部可以訪問到全局變數,而在函數外部是無法訪問到函數內部的局部變數的。
接下來,我們先看一段代碼:

let name = "水木年華";
function fn() {
    name = "張傑";
}
fn();
console.log(name);	// 張傑

由此可知,在函數內部想要修改外部的變數是十分容易的一件事. 尤其是全局變數. 這是非常危險的. 試想, 我寫了一個函數. 要用到name, 結果被別人寫的某個函數給修改掉了. 多難受.
接下來,我們來看一個例子:
比如目前有一個項目,需要兩個人一起開發,分別是小明.js和小芳.js
小明.js:

var key = '小明的key';
function jiami() {
	console.log("小明使用", key, "來加密");
}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小明.js"></script>
    <script>
        fn();   // 小明使用 小明的key 來加密
    </script>
</head>
<body>
</body>
</html>

小芳.js:

var key = '小芳的key';
function jiami() {
	console.log("小芳使用", key, "來加密");
}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小芳.js"></script>
    <script>
        fn();   // 小芳使用 小芳的key 來加密
    </script>
</head>
<body>
</body>
</html>

兩個人的代碼在各自執行的時候都沒有問題,但是當把兩個js文件同時在一個html文件中引用時,就會出現變數值被修改的問題:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小明.js"></script>
    <script src="小芳.js"></script>
    <script>
	/*兩個人都叫jiami,怎樣區分調用的是誰的方法?*/
        jiami();
		jiami();
    </script>
</head>
<body>
</body>
</html>

執行結果:
image
很明顯,兩個腳本在同一個html文件中被引用時,變數名和函數名重覆,是不是覺得讓各自的變數名和函數名不一致就可以了?如下:
小明.js:

var xiaoming_key = '小明的key';
function xiaoming_jiami() {
	console.log("小明使用", xiaoming_key, "來加密");
}

小芳.js:

var xiaofang_key = '小芳的key';
function xiaofang_jiami() {
	console.log("小芳使用", xiaofang_key, "來加密");
}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小明.js"></script>
    <script src="小芳.js"></script>
    <script>
        xiaoming_jiami();
        xiaofang_jiami();
    </script>
</head>
<body>
</body>
</html>

執行結果:
image
這樣搞確實可以實現了哈。但是問題來了,這個項目用到的變數和函數少,js腳本也少。而在真正的羡慕中,一個html文件可能引用幾十個js文件,每個js文件又有成百上千的變數或者函數,這樣的話我在一個一個的去修改變數名和函數名?不太現實,況且在html文件中引用的時候也不方便。那麼現在這種情況,我們就可以嘗試一下閉包了,比如小明和小芳之前的js代碼是在一個函數中的:
小明.js:

function ming() {
	/*
	在外面在套一層函數結構,使key變數和jiami函數的作用範圍由之前的全局變為ming函數的局部作用域
	*/
	var key = '小明的key';
	function jiami() {
		console.log("小明使用", key, "來加密");
	}
	return jiami;	// 最後在返回函數名供外部使用
}
ming_jiami = ming();	// 這裡的ming_jiami要與小芳的進行區分

小芳.js:

function fang() {
	var key = '小芳的key';
	function jiami() {
		console.log("小芳使用", key, "來加密");
	}
	return jiami;
}
fang_jiami = fang();	// 小芳這裡的fang_jiami也要與小明區分
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小明.js"></script>
    <script src="小芳.js"></script>
    <script>
        ming_jiami();    // 小明
        fang_jiami();    // 小芳
    </script>
</head>
<body>
</body>
</html>

執行結果:
image
現在執行結果確實沒有問題了,兩個人的腳本都可以正常使用了。但是兩個人外層的函數還是有可能是一樣的,我們再用之前學過的匿名函數再優化一下:
小明.js:

ming_jiami = (function(){
	var key = '小明的key';
	function jiami() {
		console.log("小芳使用", key, "來加密");
	}
	return jiami;
})();

小芳.js:

fang_jiami = (function () {
	var key = '小芳的key';
	function jiami() {
		console.log("小芳使用", key, "來加密");
	}
	return jiami;
})();
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="小明.js"></script>
    <script src="小芳.js"></script>
    <script>
        ming_jiami();    // 小明
        fang_jiami();    // 小芳
    </script>
</head>
<body>
</body>
</html>

執行結果:
image
OK,至此,就完美的優化了這個代碼。在最外面接收內部返回的參數的變數兩個人重覆的的幾率就已經非常低了。何為閉包? 上面這個就是閉包。相信你百度一下就會知道,什麼內層函數使用外層函數變數。什麼讓一個變數常駐記憶體,等等。其實你細看,它之所以稱之為閉包~,它是一個封閉的環境,在內部,自己和自己玩兒,避免了對該模塊內部的衝擊和改動,避免的變數之間的衝突問題。

閉包的特點:

  • 內層函數對外層函數變數的使用
  • 會讓變數常駐與記憶體

這倆玩意就不解釋了,和python的閉包是一個意思。不懂沒關係,能看懂他的執行過程就好。

本文來自博客園,作者:生而自由愛而無畏,轉載請註明原文鏈接:https://www.cnblogs.com/zczhaod/p/17645458.html


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

-Advertisement-
Play Games
更多相關文章
  • 簡述 上篇文章介紹瞭如何搭建 prometheus 監控體系,監控 linux 伺服器,這篇文章跟大家介紹如何監控 redis,以及我們要關註的指標都有哪些 監控 redis 需要關註什麼指標 在《聊聊監控》這篇文章,介紹了 google 提出的監控四個黃金指標(沒看過的朋友可以看看這篇文章),下麵 ...
  • 轉載請標明出處:https://www.cnblogs.com/tangZH/p/17307406.html - [kotlin協程小記](http://77blogs.com/?p=73 "kotlin協程小記") - [協程的async使用](http://77blogs.com/?p=77 " ...
  • - [kotlin協程小記](http://77blogs.com/?p=73 "kotlin協程小記") - [協程的async使用](http://77blogs.com/?p=77 "協程的async使用") - [kotlin協程異常處理之-try catch ](http://77blog ...
  • ##### CSS 背景屬性 ##### 1. background-color(背景顏色) 頁面的背景顏色有四種屬性值表示,分別是transparent(透明),RGB十進位顏色表示,十六進位顏色表示和顏色單詞表示。 屬性使用: ``` /* background-color: transpare ...
  • ##### 8 文本屬性 font-style(字體樣式風格) ``` /* 屬性值: normal:設置字體樣式為正體。預設值。 italic:設置字體樣式為斜體。這是選擇字體庫中的斜體字。 oblique:設置字體樣式為斜體。人為的使文字傾斜,而不是去使用字體庫的斜體字。 */ ``` font ...
  • ##### 5 變數提升 看以下代碼, 或多或少會有些問題的. ```javascript function fn(){ console.log(name); var name = '大馬猴'; } fn() ``` 發現問題了麽. 這麼寫代碼, 在其他語言里. 絕對是不允許的. 但是在js里. 不 ...
  • ##### 7 選擇器優先順序 所謂CSS優先順序,即是指CSS樣式在瀏覽器中被解析的先後順序。樣式表中的特殊性描述了不同規則的相對權重。 ``` /* !important > 行內樣式>ID選擇器 > 類選擇器 > 標簽 > 通配符 > 繼承 > 瀏覽器預設屬性 1 內聯樣式表的權值最高 style ...
  • ##### 6 樣式繼承 CSS的樣式表繼承指的是,特定的CSS屬性向下傳遞到子孫元素。總的來說,一個HTML文檔就是一個家族,然後html元素有兩個子元素,相當於它的兒子,分別是head和body,然後body和head各自還會有自己的兒子,最終形成了一張以下的家族譜。 ![image](http ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...