亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

Javascript中的閉包及應用

2019-11-11 03:08:56
字體:
來源:轉載
供稿:網友

閉包是javaScript比較有意思的特性,也是比較難搞懂的一個概念。

典型示例

一個比較典型的例子就是打印循環計數—— 首先我們寫一個小循環,直接打印循環變量i

function testA() { for(var i = 0; i < 10; i++) { console.log("current: " + i); }};

這個程序的輸出很簡單

current: 0current: 1current: 2current: 3current: 4current: 5current: 6current: 7current: 8current: 9

接下來做一點小改變——不再循環中立即打印變量了,而是延遲一段時間再打印(類似的是在循環中給div標簽添加onClick監聽,等到用戶點擊時再輸出變量值),這時候代碼變為:

function testB() { for(var i = 0; i < 10; i++) { setTimeout(function() { console.log("current: " + i); }, 100); }}

運行這個方法,輸出:

current: 10current: 10current: 10current: 10current: 10current: 10current: 10current: 10current: 10current: 10

顯然沒有符合預期,所有延遲的調用都輸出了循環變量i最后的值。如果機械的記憶書本上的概念就是閉包只能取得包含函數中任何變量的最后一個值。

閉包的使用

換一種思路其實不難理解。

這就好比一個工人生產一批零件,每生產和一個,他就應當在這個零件上打印一個序號。然而這個工人忘記了這道程序,又很不巧打印序號的機器很智能,每監測到生產一個零件,序號就自動加1。當這個工人完成工作以后,他忽然想起忘記打印序號的工序了,他拿起機器就往零件上打印,結果發現所有序號都一樣…… 這就十分悲劇了。要想打印上正確的序號,必須記住要每生產一個零件以后就打印,萬萬不可等完工后再來這道工序。

類似的,在setTimeout中,我們傳遞了一個回調函數延遲調用,回調函數就好比打印序號這道工序,當時沒有執行,等到執行時讀取的都是變量i,自然拿到的就是最后的值了。 所以就需要循環中每調用一次setTimeout就保存住當前的循環變量i。那這如何實現呢?

下面這段程序可以給我們一些啟示:

var num = 5;function testNum(_num){ _num = 9;}testNum(num);console.log(num); // 5

由于函數參數是按值傳遞的,傳遞給testNum的只是num的值,在函數里如何改變型參的值,是不影響原變量的。 這個特性就非常好了,既然我們想保存循環變量的每一個值,那就每循環一步,調用一個函數,把循環變量傳進去就好了。這樣我們在函數的內部,永遠拿到的是調用這個函數時型參對應原變量的值。

function testB() { for (var i = 0; i < 10; i++) { help(i); }}

然后在這個方法里再去設置延時任務

function help(num) { setTimeout(function () { console.log("current: " + num); }, 100);}function testB() { for (var i = 0; i < 10; i++) { help(i); }}

這樣程序輸出就是1~10了。

進一步優化一下,僅為了保存循環變量,就在外面聲明一個函數,非常浪費。在Javascript中更好的做法是聲明一個匿名函數,并立即調用它

function testB() { for (var i = 0; i < 10; i++) { // 匿名函數包含一個參數num (function(num) { setTimeout(function () { console.log("current: " + num); }, 100); })(i); // 立即調用了匿名函數,確保i當前值保存在閉包的環境中 }}

這種寫法不太直觀,因為我們無法直接看出循環體中做了什么——真正關鍵的setTimeout是在匿名函數中調用的,多嵌套了一層。如果在循環中直接調用setTimeout意圖就更加清晰了。按著這個思路,先將不使用匿名函數的版本做一下調整:

function help(num) { return function() { console.log("current: " + num); }}function testC() { for (var i = 0; i < 10; i++) { setTimeout(help(i), 100); // 立即執行help保存i的值 }}

由于setTimeout第一個參數回調是函數類型,所以我們需要在help方法中返回一個函數,這樣調用help后,將返回的匿名函數傳遞給setTimeout。 同樣的,在這里聲明一個單獨的函數有些浪費,再次改為匿名函數的版本:

function testC() { for (var i = 0; i < 10; i++) { setTimeout((function (num) { return function () { console.log("current: " + num); } })(i), 100); }}

setTimeout第一個參數時候,定義了一個返回匿名函數的匿名函數,并立即執行它,達到了同樣的效果。

應用案例

最后看一個實際應用的例子。 在node.js中可以使用fs.readdir函數來遍歷給定文件夾,返回的結果是文件夾中除.,..以外所有文件/文件夾的數組files。

var filePath = path.join(__dirname, "someFolderName");fs.readdir(filePath, function (err, files) { parseResult(err, files); });

繼而在parseResult中,我們想對這個數組進行處理,篩選出其中的文件夾進行進一步的操作

function parseResult(err, files) { var length = 0; for (var tmp in files) { length++; } for (var item in files) { var filePath = path.join(__dirname, "someFoladerName/" + files[item]); fs.stat(filePath, (function (num) { console.log("the num is " + num); var currentItem = num; return function (err, stats) { if (stats.isDirectory) { // 檢測到文件夾,做相應處理 } if (currentItem == (length - 1)) { // currentItem是string類型的序號,轉換后判斷 // 對所有文件/文件夾進行的stat異步調用完成后,進行最后操作 } } })(item)); // 立即調用匿名函數,保存item當前值并返回其中聲明的處理函數 }}
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲欧美日韩图片| 97超碰蝌蚪网人人做人人爽| 亚洲欧美日韩精品久久| 国产精品国内视频| 久久精品91久久香蕉加勒比| 成人免费福利视频| 欧美一区二区三区……| 欧美黑人性猛交| 九九久久综合网站| 中文字幕一区二区三区电影| 亚洲欧美国产精品va在线观看| 国产97色在线|日韩| 国产一区二区三区在线观看网站| 欧美国产高跟鞋裸体秀xxxhd| 亚洲成人999| 久久在精品线影院精品国产| 97视频在线观看免费| 久99九色视频在线观看| 国产91精品久久久久久久| 国产成人jvid在线播放| 日韩欧美一区二区三区| 在线日韩日本国产亚洲| 亚洲裸体xxxx| 96pao国产成视频永久免费| 这里只有精品视频在线| 俺去了亚洲欧美日韩| 欧美高清性猛交| 免费97视频在线精品国自产拍| 亚洲女人天堂网| 亚洲一区制服诱惑| 国产精品1区2区在线观看| 亚洲乱码av中文一区二区| 久久久人成影片一区二区三区观看| 91免费欧美精品| 日韩在线播放一区| 国产日韩换脸av一区在线观看| 97超碰蝌蚪网人人做人人爽| 欧美国产日韩二区| 成人动漫网站在线观看| 亚洲精品国产精品久久清纯直播| 最好看的2019的中文字幕视频| 亚洲精品99久久久久中文字幕| 亚洲免费人成在线视频观看| 亚洲欧美日韩高清| 国模吧一区二区三区| 91精品国产高清自在线看超| 欧美一区二区三区……| 91人人爽人人爽人人精88v| 亚洲女性裸体视频| 国产精品一区二区三| 国产成人一区二| 久久精品久久久久久| 欧美重口另类videos人妖| 欧美做爰性生交视频| 久久精品国产成人精品| 91精品在线观看视频| 日韩成人在线网站| 久久久亚洲成人| 日韩经典中文字幕在线观看| 38少妇精品导航| 亚洲国产欧美一区二区三区同亚洲| 欧美日韩国产丝袜另类| 69国产精品成人在线播放| 精品久久香蕉国产线看观看gif| 91精品国产综合久久男男| 日韩精品免费在线播放| 国产精品久久久久久久久久| 欧美香蕉大胸在线视频观看| 久久综合色影院| 69av视频在线播放| 亚洲天堂第一页| 久久深夜福利免费观看| 色狠狠av一区二区三区香蕉蜜桃| 国产美女直播视频一区| 亚洲精品久久在线| 国产国语videosex另类| 国产精品久久久久久超碰| 成人黄色午夜影院| 日本精品视频在线| 国产精品27p| 亚洲人午夜色婷婷| 欧美黄色片视频| 92国产精品视频| 色哟哟网站入口亚洲精品| 日韩在线一区二区三区免费视频| 不卡伊人av在线播放| 午夜精品久久久久久久99热浪潮| 在线视频欧美日韩| 91精品久久久久久久久中文字幕| 国产大片精品免费永久看nba| 日本aⅴ大伊香蕉精品视频| 福利视频导航一区| 91在线中文字幕| 伊人久久久久久久久久久| 精品久久久国产精品999| 国产精品美女在线观看| 久久精品久久久久久国产 免费| 国产精品亚洲精品| 日韩美女毛茸茸| 欧美在线视频网站| 91超碰caoporn97人人| 亚洲www视频| 日韩欧美国产免费播放| 欧洲精品毛片网站| 国产主播欧美精品| 欧美性视频精品| 欧美在线一级视频| 欧美猛交免费看| 91久久中文字幕| 九九久久国产精品| 青青草一区二区| 国产91精品最新在线播放| 国产成人亚洲综合青青| 国产mv久久久| 日韩在线中文字幕| www.久久久久| 精品在线小视频| 日韩av理论片| 91tv亚洲精品香蕉国产一区7ujn| 国产日韩欧美91| 中文字幕在线看视频国产欧美在线看完整| 午夜精品美女自拍福到在线| 亚洲第一中文字幕在线观看| 久久91亚洲精品中文字幕| 成人免费黄色网| 国产91精品网站| 久久精品视频在线观看| 亚洲国产精品电影| 国产一区二区三区高清在线观看| 精品久久久中文| 一区二区三区视频免费在线观看| 亚洲淫片在线视频| **欧美日韩vr在线| 亚洲毛片在线观看.| 亚洲天堂2020| 亚洲精品午夜精品| 中文字幕亚洲专区| 亚洲专区国产精品| 国产亚洲欧美日韩精品| 日韩精品在线观看一区二区| 亚洲国产成人在线视频| 国产综合色香蕉精品| 亚洲一区二区久久| 日韩一区二区三区国产| 欧美精品在线极品| 伊人伊成久久人综合网小说| 久久99精品久久久久久青青91| 日韩av电影中文字幕| 国产大片精品免费永久看nba| 精品国产91乱高清在线观看| 清纯唯美亚洲综合| 中国china体内裑精亚洲片| 国产精品一区二区av影院萌芽| 川上优av一区二区线观看| 91夜夜揉人人捏人人添红杏| 精品在线欧美视频| 456国产精品| 91精品视频大全| 国产日韩在线看| 亚洲精品国产精品国产自| 国产视频精品免费播放| 日韩在线观看精品| 亚洲精品www久久久久久广东| 黑人巨大精品欧美一区二区一视频|