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

首頁 > 編程 > JavaScript > 正文

js閉包實(shí)例匯總

2019-11-20 13:56:34
字體:
供稿:網(wǎng)友

Js閉包
閉包前要了解的知識(shí)
1. 函數(shù)作用域
(1).Js語言特殊之處在于函數(shù)內(nèi)部可以直接讀取全局變量

復(fù)制代碼 代碼如下:

<script type="text/javascript">
var n=100;
function parent(){
  alert(n);
}
parent();//100
</script>

如果在php里

復(fù)制代碼 代碼如下:

<?php
$n=100;
function parent(){
  echo $n;
}
parent();//會(huì)報(bào)錯(cuò) n未定義
?>

(2).在函數(shù)外部無法讀取函數(shù)內(nèi)的局部變量

復(fù)制代碼 代碼如下:

<script type="text/javascript">
function parent(){
  var m=50;
}
parent();
alert(m);//報(bào)錯(cuò) m未定義
</script>

注意函數(shù)內(nèi)部聲明變量時(shí)一定要加var,否則就聲明了一個(gè)全局變量

復(fù)制代碼 代碼如下:

function parent(){
m=50;
}
parent();
alert(m);//50

//當(dāng)然在php里更是如此了,

復(fù)制代碼 代碼如下:

<?php
function parent(){
  global $m;//全局 ,定義與賦值要分開
  $m=50;
}
parent();
echo $m;//50
?>
//沒global的話,一樣會(huì)報(bào)沒定義的錯(cuò)誤

有時(shí),需要得到函數(shù)內(nèi)部的的局部變量,就需要變通的方法實(shí)現(xiàn)利用js變量作用域的特點(diǎn),如在函數(shù)內(nèi)部定義子函數(shù),對(duì)于子函數(shù)來說,父函數(shù)就是它的全局,子函數(shù)可以訪問父函數(shù)里的變量(對(duì)于整個(gè)js代碼來說又是局部變量)

復(fù)制代碼 代碼如下:

<script type="text/javascript">
function parent(){
   var m=50;
   function son(){
        alert(m);
   }
   return son;
}
var s=parent();//將結(jié)果保存在全局里
s();//50
</script>

Parent內(nèi)部所有局部變量對(duì)其子函數(shù)來說都是可見的,但其子函數(shù)內(nèi)的局部變量對(duì)其父函數(shù)是不可見的,這就是js特有的鏈?zhǔn)阶饔糜蚪Y(jié)構(gòu),子對(duì)象會(huì)一級(jí)一級(jí)地向上查找所有父對(duì)象的變量,父對(duì)象的所有變量對(duì)子對(duì)象都是可見的,反之不成立!上面的son函數(shù)就是閉包
有些同學(xué)可能這樣

復(fù)制代碼 代碼如下:

function parent(){
   var m=50;
   function son(){
        alert(m);
   }
}
parent();
son()//會(huì)報(bào) 函數(shù)son未定義

注意 在javascript里,在函數(shù)里聲明的函數(shù)都是局部的,函數(shù)運(yùn)行完后就釋放了
注意這點(diǎn)與php的區(qū)別

復(fù)制代碼 代碼如下:

<?php
function parent(){
  function son(){
      $m=50;
      echo $m;
  }
}
parent();
son();//輸出50 不會(huì)報(bào)錯(cuò)
?>

閉包

函數(shù)內(nèi)部定義函數(shù),連接函數(shù)內(nèi)部和外部的橋梁
閉包的作用有2個(gè):
一是前面提到的讀取函數(shù)內(nèi)部的變量,
二是讓這些變量的值保存在內(nèi)存中,實(shí)現(xiàn)數(shù)據(jù)共享
下面是幾個(gè)閉包的例子

復(fù)制代碼 代碼如下:

<script type="text/javascript">
var cnt=(function(){
    var i=0;
    return function(){
        alert(i);
        i++;
    }
})();
cnt();//0
cnt();//1
cnt();//2
cnt();//3

</script>

把匿名函數(shù)的執(zhí)行結(jié)果(即對(duì)里面子函數(shù)的聲明賦給全局變量cut),i就保存在內(nèi)存里了
執(zhí)行cut()時(shí)就直接從內(nèi)存取值了,i只有cnt()函數(shù)才能調(diào)用,直接alert(i)是不行的
還可以向閉包內(nèi)傳參

復(fù)制代碼 代碼如下:

var cnt=(function(num){
return function(){
    alert(num);
    num++;
  }
})(5);
cnt();//5
cnt();//6
cnt();//7
//當(dāng)然還可以調(diào)用時(shí)傳參
var cnt=(function(){
    var i=0;
return function(num){
    num+=i;
    alert(num);
    i++;
  }
})();
cnt(1);//1
cnt(2);//3
cnt(3);//5

為了對(duì)閉包有更好的理解,我們看以下代碼
比如我想返回一個(gè)數(shù)組,數(shù)組里面有5個(gè)函數(shù),第一個(gè)函數(shù)彈出0,第二個(gè)彈出1... 
代碼如果這樣寫

復(fù)制代碼 代碼如下:

function box(){
  var arr=[];
  for(i=0;i<5;i++){
      arr=function(){return i;}
    }
return arr;  
}
var a=box();
alert(a);//包含五個(gè)函數(shù)體的數(shù)組
alert(a[0]());
alert(a[1]());

彈出的函數(shù)體
function(){return i;}    }
最后這個(gè)i是4,之后++成為5
For循環(huán)停止
發(fā)現(xiàn)均彈出5,明顯不符合我們的要求

解決方案1
自我即時(shí)執(zhí)行里面的函數(shù)

復(fù)制代碼 代碼如下:

function box(){
  var arr=[];
  for(i=0;i<5;i++){
      arr=(function(num){return i;})(i);
    }
return arr;  
}
var a=box();
for(var i=0;i<a.length;i++){
  alert(a);
}

但是我們發(fā)現(xiàn) 返回的數(shù)組里的元素是函數(shù)執(zhí)行的結(jié)果,但我們想要的是函數(shù)有得升級(jí)我們的代碼

解決方案2
閉包實(shí)現(xiàn)

復(fù)制代碼 代碼如下:

function box(){
var arr=[];
        for(var i=0;i<5;i++){

                 arr=(function(num){
                     return function(){return num;}
                 })(i);

         }
return arr;        
}

var arr=box();

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

    alert(arr());//0,1,2,3,4
}

關(guān)鍵代碼

復(fù)制代碼 代碼如下:

arr=(function(num){
         return function(){return num;}
})(i);


i=0 時(shí)
arr[0]=(function(num){return function(){return num;}})(0);

1時(shí)


arr[1]=(function(num){return function(){return num;}})(1); 

  以上就是閉包的好處!非常簡(jiǎn)單實(shí)用吧。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 许昌县| 宁河县| 伊宁市| 伊川县| 鄯善县| 措勤县| 罗田县| 漳州市| 曲沃县| 共和县| 苏尼特左旗| 巨野县| 福安市| 马尔康县| 赣州市| 罗田县| 乌海市| 凤台县| 疏附县| 陆川县| 怀来县| 麻栗坡县| 阿巴嘎旗| 朝阳区| 新昌县| 林芝县| 台中市| 新民市| 达孜县| 砚山县| 灌阳县| 正宁县| 巴彦淖尔市| 永康市| 清原| 阳高县| 襄垣县| 汶上县| 海兴县| 泰来县| 文昌市|