国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發(fā) > JS > 正文

JavaScript解析機(jī)制與閉包原理實(shí)例詳解

2024-05-06 16:48:51
字體:
供稿:網(wǎng)友

本文實(shí)例講述了JavaScript解析機(jī)制與閉包原理。分享給大家供大家參考,具體如下:

js解析機(jī)制:

js代碼解析之前會(huì)創(chuàng)建一個(gè)如下的詞法環(huán)境對(duì)象(倉(cāng)庫(kù)):LexicalEnvironment{ }

在掃描js代碼時(shí)會(huì)把:

1、用聲明的方式創(chuàng)建的函數(shù)的名字;

2、用var定義的變量的名字存到這個(gè)詞法環(huán)境中;

3、同名的時(shí)候:函數(shù)聲明會(huì)覆蓋變量,下面的函數(shù)聲明會(huì)覆蓋上面的同名函數(shù);

4、函數(shù)的值為:對(duì)函數(shù)的一個(gè)引用; 變量的值為undefined;

5、如果用函數(shù)表達(dá)式的方式創(chuàng)建一個(gè)函數(shù):

var fn = function(){ } 這樣詞法環(huán)境中存的是一個(gè)變量名fn,并賦值為undefined;

在調(diào)用函數(shù)的時(shí)候如果在函數(shù)上面調(diào)用就會(huì)出現(xiàn)和變量一樣的情況報(bào)錯(cuò)undefined;

這也是以兩種不同方式創(chuàng)建函數(shù)的區(qū)別;

LexicalEnvironment(這個(gè)詞法環(huán)境===window){fn: 對(duì)函數(shù)的一個(gè)引用;b:undefined;}

用聲明的方式創(chuàng)建的函數(shù):

function fn(){ };

用var定義的變量:

var b=5;

每次調(diào)用函數(shù)的時(shí)候就會(huì)創(chuàng)建一個(gè)新的詞法環(huán)境對(duì)象(倉(cāng)庫(kù)):LexicalEnvironment{ };
在解析函數(shù)內(nèi)部的變量和函數(shù)聲明的時(shí)候跟全局詞法環(huán)境相同,不過有兩點(diǎn)需要注意,如下:

LexicalEnvironment(這個(gè)詞法環(huán)境===fn){a:對(duì)函數(shù)的一個(gè)引用;(解析的時(shí)候函數(shù)聲明把變量覆蓋了,盡管變量已經(jīng)被賦值為1)b:2;(解析的時(shí)候把變量存在了詞法環(huán)境里,同時(shí)賦值為2)}

 

function fn (a,b){  alert(a)// function a(){ }  alert(b)//2  var b= 100;  function a(){ }}fn(1,2);

調(diào)用函數(shù)并傳遞參數(shù)的時(shí)候,詞法環(huán)境里會(huì)再存變量名的同時(shí)賦值,如果是函數(shù)內(nèi)部有同名的函數(shù)聲明則會(huì)把傳入的參數(shù)覆蓋;如果形參只有一個(gè),那么另一個(gè)實(shí)參則被賦值為undefined;

閉包:

定義:(有多種定義)

1、(比較通俗的定義):函數(shù)嵌套函數(shù),內(nèi)部函數(shù)可以引用外部函數(shù)的參數(shù)和變量,這些參數(shù)和變量不會(huì)被垃圾回收機(jī)制所回收;

2、在計(jì)算機(jī)科學(xué)中,閉包是詞法閉包的簡(jiǎn)稱,是引用了自由變量的函數(shù),這個(gè)被引用的自由變量將和這個(gè)函數(shù)一同存在,即使已經(jīng)離開了創(chuàng)造它的環(huán)境也不例外(意思就是不會(huì)被銷毀)。

3、閉包是由函數(shù)和其相關(guān)的引用環(huán)境組合而成的實(shí)體。(潛臺(tái)詞就是這個(gè)函數(shù)將和引用環(huán)境同時(shí)存在,必須有引用)

綜合來說,不管怎么定義都是在圍繞著兩個(gè)本質(zhì):函數(shù)在引用變量,這個(gè)變量將不會(huì)被銷毀。

什么叫做被引用的自由變量離開了創(chuàng)造它的環(huán)境?如下:

function fn(){  var a = 10;  var b = 20;  return function fn2(){    alert(a);  }}var result = fn();result();//10;

以上代碼就是fn2在被return出去以后,離開了fn函數(shù)這個(gè)環(huán)境,但是在外部調(diào)用依然能夠訪問到fn的變量;

這就是被引用的變量不會(huì)被銷毀;同理在自執(zhí)行函數(shù)這個(gè)閉包里,雖然自執(zhí)行函數(shù)在自身執(zhí)行過后內(nèi)部變量本該被垃圾回收機(jī)制所回收,但是由于其內(nèi)部有引用它的變量的子函數(shù),也就是說構(gòu)成了閉包,它的變量依然不會(huì)被銷毀;

由此可見閉包的一個(gè)作用就是:我們能夠通過閉包的方法來在外部訪問到一個(gè)內(nèi)部函數(shù)的變量;

很多人在解釋閉包的時(shí)候都會(huì)把子函數(shù)return出去以后在外部調(diào)用,其實(shí)無論在哪里調(diào)用,閉包都已經(jīng)形成了,只要是函數(shù)嵌套函數(shù),并且子函數(shù)引用了父函數(shù)的變量,(不論子函數(shù)有沒有被調(diào)用,這個(gè)用一種方法證明:在子函數(shù)內(nèi)部打斷點(diǎn),在f12中觀察閉包里的內(nèi)容,已經(jīng)出現(xiàn)了引用函數(shù),這時(shí)候調(diào)用還沒有被執(zhí)行)這個(gè)時(shí)候閉包已經(jīng)形成了。

閉包的本質(zhì):就是形成了作用域鏈。

注意:形成閉包的條件:1、函數(shù)要嵌套;2、子函數(shù)要引用父函數(shù)的變量(如果沒有引用,則不會(huì)形成閉包,如果是引用父函數(shù)的父函數(shù)的變量也會(huì)形成閉包);

父函數(shù)每調(diào)用一次,就會(huì)形成一個(gè)新的閉包(函數(shù)每調(diào)用一次,就 會(huì)復(fù)制一份),也就是說形成一個(gè)新的詞法作用域,重新引用父函數(shù)的變量;

以下代碼說明:

function fn() {  var num = 1;  return function() {    num++;    alert(num);  }}var result1 = fn();result1();//2;result1();//3;var result2 = fn();result2();//2;result2();//3;

在fn被調(diào)用兩次時(shí),都形成了新的閉包,有各自新的詞法作用域,所以result2的輸出結(jié)果不受result1的影響;

函數(shù)每調(diào)用一次,都會(huì)復(fù)制一份新的,可以說明for循環(huán)里i的問題;

for (var i = 0; i < 5; i++) {  (function(index){    oDiv.onclick = function() {      alert(index)    }  })(i)}

自執(zhí)行函數(shù)每調(diào)用一次都會(huì)復(fù)制一份新的,傳進(jìn)的i值也在變化,由于在函數(shù)靜態(tài)作用域里,在預(yù)解析階段已經(jīng)確定了變量的作用域,所以子函數(shù)引用的父函數(shù)變量index只能是每次復(fù)制的那個(gè)父函數(shù)變量,所以就實(shí)現(xiàn)了我們想要取不同的i值的目的;

閉包的用途:

1、匿名自執(zhí)行函數(shù)

不污染全局變量,(否則被聲明為全局變量的話別的函數(shù)可能誤用這些變量;造成全局對(duì)象過于龐大,影響訪問速度(因?yàn)樽兞康娜≈凳切枰獜脑玩溕媳闅v的)。提高效率;

2、結(jié)果緩存:

我們開發(fā)中會(huì)碰到很多情況,設(shè)想我們有一個(gè)處理過程很耗時(shí)的函數(shù)對(duì)象,每次調(diào)用都會(huì)花費(fèi)很長(zhǎng)時(shí)間利用閉包,它不會(huì)釋放外部的引用,從而函數(shù)內(nèi)部的值可以得以保留。,這樣我們?cè)诘诙握{(diào)用的時(shí)候,就會(huì)從緩存中讀取到該資源。

3、實(shí)現(xiàn)封裝;

4、實(shí)現(xiàn)類和繼承(構(gòu)造函數(shù));

希望本文所述對(duì)大家JavaScript程序設(shè)計(jì)有所幫助。


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到JavaScript/Ajax教程頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 嵊州市| 永定县| 彰化县| 信宜市| 昭平县| 襄汾县| 阜宁县| 拉萨市| 左云县| 雅安市| 高平市| 高青县| 南溪县| 敦化市| 山东| 平和县| 小金县| 兴海县| 青铜峡市| 太原市| 雷山县| 崇左市| 漳浦县| 湘西| 汤阴县| 玉山县| 富顺县| 三门县| 深州市| 延长县| 长乐市| 北海市| 湘阴县| 图木舒克市| 凤山县| 日喀则市| 庆安县| 汉阴县| 湖州市| 西乌| 沂南县|