函數(shù)的類型可以是直接聲明一個(gè)函數(shù),或者匿名來聲明一個(gè)函數(shù)。普通函數(shù)和匿名函數(shù)基本來說是相同的,但是普通函數(shù)會(huì)自動(dòng)把函數(shù)名提到作用域的最前面。
普通函數(shù)
function a(){
};
匿名函數(shù)
var a = function(){
};
沒有名字的函數(shù):function( ){ };
使用function關(guān)鍵字聲明一個(gè)函數(shù),但未給函數(shù)命名,就叫做匿名函數(shù)。立即執(zhí)行函數(shù)(function(){...})()內(nèi)部定義的變量不會(huì)和外部的變量發(fā)生沖突。
閉包
閉包的含義:閉包也就是說函數(shù)的嵌套,可使內(nèi)部訪問外部作用域。內(nèi)存常駐,變量持久化。
function aaa(){
var a=5;
function bbb(){
alert(a);
}
bbb{};
}
這就是一個(gè)簡(jiǎn)單的閉包,也是對(duì)閉包最簡(jiǎn)單的理解。當(dāng)然閉包還有更深入的理解,比如執(zhí)行環(huán)境、活動(dòng)對(duì)象以及作用域等等。只是希望大家能對(duì)此有一個(gè)基本的了解。
閉包的缺點(diǎn): 函數(shù)執(zhí)行完畢之后,會(huì)自動(dòng)銷毀,沒有立即銷毀的函數(shù),會(huì)產(chǎn)生內(nèi)存泄漏。
在使用中最好不要刻意的使用閉包。
習(xí)題
function fun(n,o){
console.log(o);
return{
fun:function(m){
return fun(m,n);
}
};
}
var a=fun(0);a.fun(1);a.fun(2);a.fun(3);
var b=fun(0).fun(1).fun(2).fun(3);
var c=fun(0).fun(1);c.fun(2);c.fun(3);
var a 的答案為undefined,0,0,0
fun(0)調(diào)用了第一層fun函數(shù),n=1,o則為undefined;fun(1)時(shí)m=1,fun閉包了外層函數(shù)的n,第一次調(diào)用的n=0,即m=1,并在內(nèi)部調(diào)用了第一層fun函數(shù)fun(1,0),所以o為0;fun(2)時(shí)m=2,但還是調(diào)用了a.fun,所以內(nèi)部還是調(diào)用了第一層的fun(2,0),所以o為0;fun(3)同理,依然為0;
var b 的答案為undefined,0,1,2
fun(0)還是調(diào)用了第一層fun函數(shù),o依然為undefined;fun(1)時(shí)m=1,fun閉包了外部函數(shù)的n,即m=1,n=0,o則為0;fun(2)m=2 ,此時(shí)當(dāng)前的fun函數(shù)不是第一次執(zhí)行的返回對(duì)象,而是第二次執(zhí)行的返回對(duì)象。而在第二次執(zhí)行第一層fun函數(shù)時(shí)(1,0)所以n=1,o=0,返回時(shí)閉包了第二次的n,在第三次調(diào)用第三層fun函數(shù)時(shí)m=2,n=1,即調(diào)用第一層fun函數(shù)fun(2,1),所以o為1;fun(3) 時(shí)m=3,閉包了第三次調(diào)用的n,同理,最終調(diào)用第一層fun函數(shù)為fun(3,2),所以o為2;
var c 的答案為undefined,0,1,1
fun(0)的o依然為undefined;fun(1) 時(shí)m為1,此時(shí)fun閉包了外層函數(shù)的n,也就是第一次調(diào)用的n=0,即m=1,n=0,并在內(nèi)部調(diào)用第一層fun函數(shù)fun(1,0),所以o為0;fun(2)時(shí)m為2,此時(shí)fun閉包的是第二次調(diào)用的n=1,即m=2,n=1,并在內(nèi)部調(diào)用第一層fun函數(shù)fun(2,1),所以o為1;fun(3)同理,但依然調(diào)用第二次的返回值,最終調(diào)用第一層fun函數(shù)fun(3,1),所以o為1;