日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

深入浅出Javascript闭包

發布時間:2025/3/21 java 30 豆豆
生活随笔 收集整理的這篇文章主要介紹了 深入浅出Javascript闭包 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

深入淺出Javascript閉包

一、引子

閉包(closure)是 JavaScript 語言的一個難點,面試時常被問及,也是它的特色,很多高級應用都要依靠閉包實現。本文盡可能用簡單易懂的話,講清楚閉包的概念、形成條件及其常見的面試題。

我們先來看一個例子:

var n = 999; function f1() { console.log(n); } f1() // 999

上面代碼中,函數f1可以讀取全局變量n。但是,函數外部無法讀取函數內部聲明的變量。

function f1() { var n = 999; } console.log(n) // Uncaught ReferenceError: n is not defined

上面代碼中,函數f1內部聲明的變量n,函數外是無法讀取的。

如果有時需要得到函數內的局部變量。正常情況下,這是辦不到的,只有通過變通方法才能實現。那就是在函數的內部,再定義一個函數。

function f1() { var n = 999; function f2() {console.log(n); // 999} }

上面代碼中,函數f2就在函數f1內部,這時f1內部的所有局部變量,對f2都是可見的。既然f2可以讀取f1的局部變量,那么只要把f2作為返回值,我們不就可以在f1外部讀取它的內部變量了嗎!

二、閉包是什么

我們可以對上面代碼進行如下修改:

function f1(){var a = 999;function f2(){console.log(a);}return f2; // f1返回了f2的引用}var result = f1(); // result就是f2函數了result(); // 執行result,全局作用域下沒有a的定義,//但是函數閉包,能夠把定義函數的時候的作用域一起記住,輸出999

上面代碼中,函數f1的返回值就是函數f2,由于f2可以讀取f1的內部變量,所以就可以在外部獲得f1的內部變量了。

閉包就是函數f2,即能夠讀取其他函數內部變量的函數。由于在JavaScript語言中,只有函數內部的子函數才能讀取內部變量,因此可以把閉包簡單理解成“定義在一個函數內部的函數”。

閉包最大的特點,就是它可以“記住”誕生的環境,比如f2記住了它誕生的環境f1,所以從f2可以得到f1的內部變量。在本質上,閉包就是將函數內部和函數外部連接起來的一座橋梁。

那到底什么是閉包呢?

當函數可以記住并訪問所在的詞法作用域,即使函數是在當前詞法作用域之外執行,這就產生了閉包。 ----《你不知道的Javascript上卷》

我個人理解,閉包就是函數中的函數(其他語言不能函數再套函數),里面的函數可以訪問外面函數的變量,外面的變量的是這個內部函數的一部分。

閉包形成的條件

函數嵌套

內部函數引用外部函數的局部變量

三、閉包的特性

每個函數都是閉包,每個函數天生都能夠記憶自己定義時所處的作用域環境。把一個函數從它定義的那個作用域,挪走,運行。這個函數居然能夠記憶住定義時的那個作用域。不管函數走到哪里,定義時的作用域就帶到了哪里。接下來我們用兩個例子來說明這個問題:

//例題1 var inner; function outer(){ var a=250; inner=function(){ alert(a);//這個函數雖然在外面執行,但能夠記憶住定義時的那個作用域,a是250} } outer(); var a=300; inner();//一個函數在執行的時候,找閉包里面的變量,不會理會當前作用域。 //例題2 function outer(x){function inner(y){console.log(x+y);} return inner; } var inn=outer(3);//數字3傳入outer函數后,inner函數中x便會記住這個值 inn(5);//當inner函數再傳入5的時候,只會對y賦值,所以最后彈出8

四、閉包的內存泄漏

棧內存提供一個執行環境,即作用域,包括全局作用域和私有作用域,那他們什么時候釋放內存的?

全局作用域----只有當頁面關閉的時候全局作用域才會銷毀

私有的作用域----只有函數執行才會產生

一般情況下,函數執行會形成一個新的私有的作用域,當私有作用域中的代碼執行完成后,我們當前作用域都會主動的進行釋放和銷毀。但當遇到函數執行返回了一個引用數據類型的值,并且在函數的外面被一個其他的東西給接收了,這種情況下一般形成的私有作用域都不會銷毀。

如下面這種情況:

function fn(){ var num=100; return function(){} } var f=fn();//fn執行形成的這個私有的作用域就不能再銷毀了

也就是像上面這段代碼,fn函數內部的私有作用域會被一直占用的,發生了內存泄漏。所謂內存泄漏指任何對象在您不再擁有或需要它之后仍然存在。閉包不能濫用,否則會導致內存泄露,影響網頁的性能。閉包使用完了后,要立即釋放資源,將引用變量指向null。

接下來我們看下有關于內存泄漏的一道經典面試題:

function outer(){var num=0;//內部變量return function add(){//通過return返回add函數,就可以在outer函數外訪問了num++;//內部函數有引用,作為add函數的一部分了console.log(num);};}var func1=outer();func1();//實際上是調用add函數, 輸出1func1();//輸出2 因為outer函數內部的私有作用域會一直被占用var func2=outer();func2();// 輸出1 每次重新引用函數的時候,閉包是全新的。func2();// 輸出2

五、閉包的作用

1.可以讀取函數內部的變量。

2.可以使變量的值長期保存在內存中,生命周期比較長。因此不能濫用閉包,否則會造成網頁的性能問題

3.可以用來實現js模塊。

js模塊:具有特定功能的js文件,將所有的數據和功能都封裝在一個函數內部(私有的),只向外暴露一個包信n個方法的對象或函數,模塊的使用者,只需要通過模塊暴露的對象調用方法來實現對應的功能。

具體請看下面的例子:

//index.html文件 <script type="text/javascript" src="myModule.js"></script> <script type="text/javascript">myModule2.doSomething()myModule2.doOtherthing() </script> //myModule.js文件 (function () {var msg = 'Beijing'//私有數據//操作數據的函數function doSomething() {console.log('doSomething() '+msg.toUpperCase())}function doOtherthing () {console.log('doOtherthing() '+msg.toLowerCase())}//向外暴露對象(給外部使用的兩個方法)window.myModule2 = {doSomething: doSomething,doOtherthing: doOtherthing} })()

六、閉包的運用

我們要實現這樣的一個需求: 點擊某個按鈕, 提示"點擊的是第n個按鈕",此處我們先不用事件代理:

..... <button>測試1</button> <button>測試2</button> <button>測試3</button> <script type="text/javascript">var btns = document.getElementsByTagName('button')for (var i = 0; i < btns.length; i++) {btns[i].onclick = function () {console.log('第' + (i + 1) + '個')}} </script>

萬萬沒想到,點擊任意一個按鈕,后臺都是彈出“第四個”,這是因為i是全局變量,執行到點擊事件時,此時i的值為3。那該如何修改,最簡單的是用let聲明i

for (let i = 0; i < btns.length; i++) {btns[i].onclick = function () {console.log('第' + (i + 1) + '個')} }

另外我們可以通過閉包的方式來修改:

for (var i = 0; i < btns.length; i++) {(function (j) {btns[j].onclick = function () {console.log('第' + (j + 1) + '個')}})(i) }

總結

以上是生活随笔為你收集整理的深入浅出Javascript闭包的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 亚洲精品无amm毛片 国内一区二区三区 | 欧美在线不卡 | 欧美亚洲另类小说 | 狠狠操天天射 | 四虎免费久久 | 国产无人区码熟妇毛片多 | 毛片网站在线看 | 日韩少妇 | 日本黄色免费网址 | 久久婷婷综合色丁香五月 | 黄色香港三级三级三级 | 深喉口爆一区二区三区 | 香蕉av777xxx色综合一区 | 一二三毛片| 99精品一区二区三区 | 国产精品熟妇人妻g奶一区 a少妇 | 99精品视频在线免费观看 | 亚洲九九色| 丁香四月婷婷 | 夜色福利 | 久久久久亚洲av成人无码电影 | 国产精品中文字幕在线观看 | 婷婷久久综合 | av在线不卡播放 | www夜片内射视频日韩精品成人 | 青青国产精品 | 能直接看的av网站 | 婷婷深爱| 亚洲中文字幕一区二区在线观看 | 亚洲午夜精品久久久久久人妖 | 久久人妻精品白浆国产 | 操操操干干干 | 亚洲欧洲自拍偷拍 | 美日韩一区 | 国产自产视频 | 中日韩毛片| 欧美成人69| 美女三级黄色 | 亚洲大尺度在线 | 亚洲老女人视频 | 国产黑丝精品 | 人体裸体bbbbb欣赏 | 黄色成人免费网站 | 99热精品久久| 亚洲一区二区91 | www日日| 黄色激情毛片 | 在线免费观看日韩av | 日韩一级 | 一级性生活免费视频 | 亚洲精品1区2区3区 国产丝袜网站 | 国产精品福利视频 | 高清日韩欧美 | 国产精品丝袜在线观看 | 日韩一级片中文字幕 | 国产中文字幕二区 | 精品国精品国产自在久不卡 | 国产在线视频二区 | 国产精美视频 | 色哟哟日韩精品 | 国产亚洲精品久久久 | 美女狠狠干 | 国产精品三级在线 | 亚洲av高清一区二区三区 | 国产一区二区中文字幕 | 日本涩涩网站 | 成人福利午夜 | 久久精品视频99 | 四色成人av永久网址 | 北条麻妃在线一区二区 | 成人视频在线观看 | 国产成人精品无码免费看在线 | www.久久久| www日日| 72种无遮挡啪啪的姿势 | 性生活网址 | h亚洲| 亚洲三级av | 乱淫67194| 天天噜 | 亚洲久久久 | 久久久久黄 | 亚洲4p| 日本激情一区二区三区 | 国产精品免费视频一区二区三区 | 国产午夜精品一区二区三区欧美 | 久热最新视频 | 国产小视频在线 | 99riav视频 | 国产成人精品久久二区二区 | 2019中文字幕在线视频 | 99成人在线视频 | 中出av在线 | 黄色成人av在线 | 四虎影院永久地址 | 欧美日本在线播放 | 久久国产精品系列 | 朝桐光在线观看 | 欧美日韩少妇 |