中文字幕免费精品_亚洲视频自拍_亚洲综合国产激情另类一区_色综合咪咪久久

深入理解javascript作用域和閉包
來源:易賢網 閱讀:959 次 日期:2016-06-15 09:44:53
溫馨提示:易賢網小編為您整理了“深入理解javascript作用域和閉包”,方便廣大網友查閱!

作用域

作用域是一個變量和函數的作用范圍,javascript中函數內聲明的所有變量在函數體內始終是可見的,在javascript中有全局作用域和局部作用域,但是沒有塊級作用域,局部變量的優先級高于全局變量,通過幾個示例來了解下javascript中作用域的那些“潛規則”(這些也是在前端面試中經常問到的問題)。

1. 變量聲明提前

示例1:

var scope=global;

function scopetest(){

console.log(scope);

var scope=local

}

scopetest(); //undefined

此處的輸出是undefined,并沒有報錯,這是因為在前面我們提到的函數內的聲明在函數體內始終可見,上面的函數等效于:

var scope=global;

function scopetest(){

var scope;

console.log(scope);

scope=local

}

scopetest(); //local

注意,如果忘記var,那么變量就被聲明為全局變量了。

2. 沒有塊級作用域

和其他我們常用的語言不同,在javascript中沒有塊級作用域:

function scopetest() {

var scope = {};

if (scope instanceof object) {

var j = 1;

for (var i = 0; i < 10; i++) {

//console.log(i);

}

console.log(i); //輸出10

}

console.log(j);//輸出1

}

在javascript中變量的作用范圍是函數級的,即在函數中所有的變量在整個函數中都有定義,這也帶來了一些我們稍不注意就會碰到的“潛規則”:

>

var scope = hello;

function scopetest() {

console.log(scope);//①

var scope = no;

console.log(scope);//②

}

在①處輸出的值竟然是undefined,簡直喪心病狂啊,我們已經定義了全局變量的值啊,這地方不應該為hello嗎?其實,上面的代碼等效于:

var scope = hello;

function scopetest() {

var scope;

console.log(scope);//①

scope = no;

console.log(scope);//②

}

聲明提前、全局變量優先級低于局部變量,根據這兩條規則就不難理解為什么輸出undefined了。

作用域鏈

在javascript中,每個函數都有自己的執行上下文環境,當代碼在這個環境中執行時,會創建變量對象的作用域鏈,作用域鏈是一個對象列表或對象鏈,它保證了變量對象的有序訪問。

作用域鏈的前端是當前代碼執行環境的變量對象,常被稱之為“活躍對象”,變量的查找會從第一個鏈的對象開始,如果對象中包含變量屬性,那么就停止查找,如果沒有就會繼續向上級作用域鏈查找,直到找到全局對象中:

作用域鏈的逐級查找,也會影響到程序的性能,變量作用域鏈越長對性能影響越大,這也是我們盡量避免使用全局變量的一個主要原因。

閉包

基礎概念

作用域是理解閉包的一個前提,閉包是指在當前作用域內總是能訪問外部作用域中的變量。

function createclosure(){

var name = jack;

return {

setstr:function(){

name = rose;

},

getstr:function(){

return name + :hello;

}

}

}

var builder = new createclosure();

builder.setstr();

console.log(builder.getstr()); //rose:hello

上面的示例在函數中返回了兩個閉包,這兩個閉包都維持著對外部作用域的引用,因此不管在哪調用總是能夠訪問外部函數中的變量。在一個函數內部定義的函數,會將外部函數的活躍對象添加到自己的作用域鏈中,因此上面實例中通過內部函數能夠訪問外部函數的屬性,這也是javascript模擬私有變量的一種方式。

注意:由于閉包會額外的附帶函數的作用域(內部匿名函數攜帶外部函數的作用域),因此,閉包會比其它函數多占用些內存空間,過度的使用可能會導致內存占用的增加。

閉包中的變量

在使用閉包時,由于作用域鏈機制的影響,閉包只能取得內部函數的最后一個值,這引起的一個副作用就是如果內部函數在一個循環中,那么變量的值始終為最后一個值。

//該實例不太合理,有一定延遲因素,此處主要為了說明閉包循環中存在的問題

function timemanage() {

for (var i = 0; i < 5; i++) {

settimeout(function() {

console.log(i);

},1000)

};

}

上面的程序并沒有按照我們預期的輸入1-5的數字,而是5次全部輸出了5。再來看一個示例:

function createclosure(){

var result = [];

for (var i = 0; i < 5; i++) {

result[i] = function(){

return i;

}

}

return result;

}

調用createclosure()[0]()返回的是5,createclosure()[4]()返回值仍然是5。通過以上兩個例子可以看出閉包在帶有循環的內部函數使用時存在的問題:因為每個函數的作用域鏈中都保存著對外部函數(timemanage、createclosure)的活躍對象,因此,他們都引用著同一變量i,當外部函數返回時,此時的i值為5,所以內部的每個函數i的值也為5。

那么如何解決這個問題呢?我們可以通過匿名包裹器(匿名自執行函數表達式)來強制返回預期的結果:

9

function timemanage() {

for (var i = 0; i < 5; i++) {

(function(num) {

settimeout(function() {

console.log(num);

}, 1000);

})(i);

}

}

或者在閉包匿名函數中再返回一個匿名函數賦值:

function timemanage() {

for (var i = 0; i < 10; i++) {

settimeout((function(e) {

return function() {

console.log(e);

}

})(i), 1000)

}

}

//timemanager();輸出1,2,3,4,5

function createclosure() {

var result = [];

for (var i = 0; i < 5; i++) {

result[i] = function(num) {

return function() {

console.log(num);

}

}(i);

}

return result;

}

//createclosure()[1]()輸出1;createclosure()[2]()輸出2

無論是匿名包裹器還是通過嵌套匿名函數的方式,原理上都是由于函數是按值傳遞,因此會將變量i的值復制給實參num,在匿名函數的內部又創建了一個用于返回num的匿名函數,這樣每個函數都有了一個num的副本,互不影響了。

閉包中的this

在閉包中使用this時要特別注意,稍微不慎可能會引起問題。通常我們理解this對象是運行時基于函數綁定的,全局函數中this對象就是window對象,而當函數作為對象中的一個方法調用時,this等于這個對象(todo 關于this做一次整理)。由于匿名函數的作用域是全局性的,因此閉包的this通常指向全局對象window:

9

var scope = global;

var object = {

scope:local,

getscope:function(){

return function(){

return this.scope;

}

}

}

調用object.getscope()()返回值為global而不是我們預期的local,前面我們說過閉包中內部匿名函數會攜帶外部函數的作用域,那為什么沒有取得外部函數的this呢?每個函數在被調用時,都會自動創建this和arguments,內部匿名函數在查找時,搜索到活躍對象中存在我們想要的變量,因此停止向外部函數中的查找,也就永遠不可能直接訪問外部函數中的變量了??傊?,在閉包中函數作為某個對象的方法調用時,要特別注意,該方法內部匿名函數的this指向的是全局變量。

幸運的是我們可以很簡單的解決這個問題,只需要把外部函數作用域的this存放到一個閉包能訪問的變量里面即可:

var scope = global;

var object = {

scope:local,

getscope:function(){

var that = this;

return function(){

return that.scope;

}

}

}

object.getscope()()返回值為local。

內存與性能

由于閉包中包含與函數運行期上下文相同的作用域鏈引用,因此,會產生一定的負面作用,當函數中活躍對象和運行期上下文銷毀時,由于必要仍存在對活躍對象的引用,導致活躍對象無法銷毀,這意味著閉包比普通函數占用

更多信息請查看腳本欄目
易賢網手機網站地址:深入理解javascript作用域和閉包
由于各方面情況的不斷調整與變化,易賢網提供的所有考試信息和咨詢回復僅供參考,敬請考生以權威部門公布的正式信息和咨詢為準!

2026上岸·考公考編培訓報班

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關于我們 | 聯系我們 | 人才招聘 | 網站聲明 | 網站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
云南網警備案專用圖標
聯系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權所有:易賢網
云南網警報警專用圖標
中文字幕免费精品_亚洲视频自拍_亚洲综合国产激情另类一区_色综合咪咪久久
国产欧美一区二区色老头| 伊大人香蕉综合8在线视| 国产精品久久久久9999| 精品1区2区3区4区| 欧美精品久久99| 午夜精品成人在线视频| 影音欧美亚洲| 欧美日韩色婷婷| 欧美在线观看视频| 一区二区三区成人| 国产无一区二区| 欧美极品在线播放| 午夜精品久久久久久久99水蜜桃| 国产精品久久综合| 欧美sm重口味系列视频在线观看| 日韩视频在线一区二区| 国产一区视频网站| 欧美日韩在线三级| 久久精品一区| 亚洲欧美另类在线| 亚洲欧洲日本国产| 欧美日韩一区在线播放| 久久久999精品| 久久精品国产精品亚洲综合| 一本一本久久| 99精品热视频只有精品10| 国内精品视频久久| 国产精品日韩电影| 国产精品卡一卡二| 国产精品久久久久久久电影| 欧美三级乱码| 国产精品毛片一区二区三区| 欧美精品一区二区三区蜜臀| 午夜久久影院| 欧美在线综合视频| 美女被久久久| 欧美日韩精品一区视频 | 国产精品你懂得| 国产精品v欧美精品∨日韩| 欧美日韩国产色站一区二区三区| 欧美伦理在线观看| 国产农村妇女精品| 好看的av在线不卡观看| 亚洲黄色在线观看| 亚洲一区二区三区免费观看| 久久精品三级| 欧美绝品在线观看成人午夜影视| 国产精品萝li| 亚洲精品中文字| 午夜一区二区三区不卡视频| 久久青草欧美一区二区三区| 日韩亚洲欧美一区| 国产一区二区主播在线| 老司机aⅴ在线精品导航| 蜜桃久久av一区| 欧美另类综合| 国产日韩成人精品| 狠狠久久亚洲欧美| 一区二区三区国产盗摄| 欧美亚洲一区| 国产精品久久99| 国产尤物精品| 亚洲香蕉在线观看| 欧美成人亚洲成人日韩成人| 国产手机视频一区二区| 91久久久久久久久久久久久| 久久精品二区三区| 国产精品v欧美精品v日本精品动漫 | 亚洲欧美成人网| 男人天堂欧美日韩| 国产精品伊人日日| 欧美中日韩免费视频| 欧美理论大片| 国产日韩欧美中文在线播放| 亚洲激情国产精品| 亚洲一区网站| 国产精品久久久久久久久果冻传媒 | 国产精品视频yy9099| 亚洲一二三区视频在线观看| 欧美一区视频| 亚洲成色www8888| 亚洲精品专区| 欧美在线观看日本一区| 国产午夜精品全部视频在线播放| 欧美午夜精品久久久久久超碰| 亚洲国产综合在线| 欧美国产欧美综合| 亚洲网址在线| 亚洲国产另类 国产精品国产免费| 18成人免费观看视频| 国产拍揄自揄精品视频麻豆| 久久精品国产清自在天天线| 欧美—级在线免费片| 国产精品高潮呻吟| 欧美久久久久免费| 欧美午夜寂寞影院| 亚洲韩国青草视频| 国产精品久久久久久久久借妻| 很黄很黄激情成人| 亚洲一级黄色片| 99re6这里只有精品| 久久久久国产免费免费| 亚洲一区激情| 亚洲最新在线视频| 亚洲精品中文字幕女同| 性娇小13――14欧美| 日韩一级黄色片| 久久国产精品毛片| 狂野欧美一区| 久久一区二区三区国产精品| 久久精品国产亚洲5555| 精品成人在线视频| 久久综合五月天婷婷伊人| 欧美激情偷拍| 99re6这里只有精品视频在线观看| 午夜亚洲视频| 一本久道久久综合狠狠爱| 国产精品ⅴa在线观看h| 久久久蜜桃精品| 久久久久久9999| 欧美日韩网站| 国产情人综合久久777777| 国产精品在线看| 国产精品美女久久久久久2018| 欧美激情精品久久久久久| 久久精品99国产精品酒店日本| 国产中文一区| 亚洲黑丝在线| 亚洲免费在线看| 麻豆av一区二区三区| 欧美激情视频给我| 欧美婷婷久久| 国产免费观看久久| 亚洲精品在线观看免费| 午夜在线精品偷拍| 欧美亚洲免费| 国产精品一区二区你懂得| aⅴ色国产欧美| 欧美国产欧美亚洲国产日韩mv天天看完整| 欧美午夜电影网| 久久久国产亚洲精品| 狠狠色综合播放一区二区| 亚洲人成绝费网站色www| 欧美日韩一区二区免费在线观看| 国产视频一区在线观看| 亚洲图片在区色| 欧美偷拍一区二区| 亚洲欧美日韩天堂一区二区| 欧美日韩在线影院| 正在播放亚洲| 亚洲精品久久久久久久久| 日韩午夜在线观看视频| 国内精品免费午夜毛片| 国产精品羞羞答答xxdd| 欧美日韩中文精品| 欧美午夜三级| 国产精品magnet| 国产精品一区一区三区| 国产午夜精品久久久久久免费视| 国产精品久久久久久久久久三级| 欧美精品在线网站| 欧美日韩免费一区| 国产精品卡一卡二| 国产色视频一区| 狠狠色2019综合网| 亚洲娇小video精品| 亚洲日本一区二区| 亚洲免费人成在线视频观看| 午夜一区二区三视频在线观看 | 亚洲欧美日韩精品综合在线观看| 亚洲视频免费观看| 欧美中文字幕视频| 欧美精品在线播放| 国产亚洲一区二区三区在线观看 | 亚洲一区二区影院| 欧美伊久线香蕉线新在线| 免费观看日韩| 国产女人水真多18毛片18精品视频| 国产一区二区你懂的| 一区二区欧美在线| 美女主播一区| 国产日本精品| 一本一本久久a久久精品综合麻豆 一本一本久久a久久精品牛牛影视 | 欧美午夜视频一区二区| 国产精品久久久久aaaa| 狠狠色综合色区| 亚洲国产精品视频一区| 亚洲男女自偷自拍| 国产精品久久久久9999| 在线观看成人网| 久久久不卡网国产精品一区| 欧美人与禽猛交乱配视频| 国产一区二区三区日韩| 亚洲一区二区三区欧美 | 99视频+国产日韩欧美| 久久www免费人成看片高清| 亚洲综合999| 欧美一区二区三区久久精品茉莉花 | 欧美日韩成人在线视频| 鲁大师影院一区二区三区|