首頁 >web前端 >js教程 >淺析Javascript匿名函數與自執行函數_javascript技巧

淺析Javascript匿名函數與自執行函數_javascript技巧

WBOY
WBOY原創
2016-05-16 15:16:001125瀏覽

函數是JavaScript中最靈活的一種對象,這裡只是講解其匿名函數的用途。匿名函數:就是沒有函數名的函數。

函數的定義,大致可分為三種方式:

第一種:這也是最常規的一種

function double(x){ 
return 2 * x; 
}

第二種:這種方法使用了Function建構函數,把參數列表和函數體都當作字串,很不方便,不建議使用。

var double = new Function('x', 'return 2 * x;');

第三種:

var double = function(x) { return 2* x; }

注意「=」右邊的函數就是一個匿名函數,創造完畢函數後,又將該函數賦給了變數square。

匿名函數的建立

第一種方式:就是上面所講的定義square函數,這也是最常用的方式之一。

第二種方式:

(function(x, y){ 
alert(x + y); 
})(2, 3);

這裡建立了一個匿名函數(在第一個括號內),第二個括號用來呼叫這個匿名函數,並傳入參數。括號是表達式,是表達式就有回傳值,所以可以在後面加一對括號讓它們執行.

自執行的匿名函數

1. 什麼是自執行的匿名函數?

它是指形如這樣的函數: (function {// code})();

2. 疑問

為什麼(function {// code})();可以執行, 而function {// code}();卻會報錯?

3. 分析

(1). 首先, 要清楚兩者的差別:
(function {// code})是表達式, function {// code}是函數宣告.
(2). 其次, js"預編譯"的特點:
js在"預編譯"階段, 會解釋函數宣告, 但卻會忽略表式.
(3). 當js執行到function() {//code}();時, 由於function() {//code}在"預編譯"階段已經被解釋過, js會跳過function(){/ /code}, 試圖去執行();, 故會報錯誤;
當js執行到(function {// code})();時, 由於(function {// code})是表達式, js會去對它求解得到返回值, 由於返回值是一個函數, 故而遇到();時, 便會被執行.

另外, 函數轉換為表達式的方法不一定要靠分組運算符(),我們還可以用void運算符,~運算符,!運算子…

如:

!function(){ 
alert("另类的匿名函数自执行"); 
}();

匿名函數與閉包

閉包的英文單字是closure,這是JavaScript中非常重要的一部分知識,因為使用閉包可以大幅減少我們的程式碼量,使我們的程式碼看起來更加清晰等等,總之功能十分強大。

閉包的意思:閉包說白了就是函數的嵌套,內層的函數可以使用外層函數的所有變量,即使外層函數已經執行完畢(這點涉及JavaScript作用域鏈)。

function checkClosure(){ 
var str = 'rain-man'; 
setTimeout( 
function(){ alert(str); } //这是一个匿名函数 
, 2000); 
} 
checkClosure();

這個例子看起來十分的簡單,仔細分析下它的執行過程還是有許多知識點的:checkClosure函數的執行是瞬間的(也許用時只是0.00001毫秒),在checkClosure的函數體內創建了一個變量str ,在checkClosure執行完畢之後str並沒有被釋放,這是因為setTimeout內的匿名函數存在這對str的引用。待到2秒後函數體內的匿名函數被執行完畢,str才被釋放。

用閉包來最佳化程式碼:

function forTimeout(x, y){ 
alert(x + y); 
} 
function delay(x , y , time){ 
setTimeout('forTimeout(' + x + ',' + y + ')' , time); 
} 
/** 
* 上面的delay函数十分难以阅读,也不容易编写,但如果使用闭包就可以让代码更加清晰 
* function delay(x , y , time){ 
* setTimeout( 
* function(){ 
* forTimeout(x , y) 
* } 
* , time); 
* } 
*/

匿名函數最大的用途是建立閉包(這是JavaScript語言的特性之一),也可以建構命名空間,以減少全域變數的使用。

var oEvent = {}; 
(function(){ 
var addEvent = function(){ /*代码的实现省略了*/ }; 
function removeEvent(){} 

oEvent.addEvent = addEvent; 
oEvent.removeEvent = removeEvent; 
})();

在這段程式碼中函數addEvent和removeEvent都是局部變量,但我們可以透過全域變數oEvent使用它,這大大減少了全域變數的使用,增強了網頁的安全性。

我們要想使用此段程式碼:

oEvent.addEvent(document.getElementById('box') , 'click' , function(){});
var rainman = (function(x , y){ 
return x + y; 
})(2 , 3); 
/** 
* 也可以写成下面的形式,因为第一个括号只是帮助我们阅读,但是不推荐使用下面这种书写格式。 
* var rainman = function(x , y){ 
* return x + y; 
* }(2 , 3);

在這裡我們建立了一個變數rainman,並透過直接呼叫匿名函數初始化為5,這個小技巧有時十分實用。

var outer = null; 
(function(){ 
var one = 1; 
function inner (){ 
one += 1; 
alert(one); 
} 
outer = inner; 
})(); 
outer(); //2 
outer(); //3 
outer(); //4

這段程式碼中的變數one是一個局部變數(因為它被定義在一個函數之內),因此外部是不可以存取的。但這裡我們建立了inner函數,inner函數是可以存取變數one的;又將全域變數outer引用了inner,所以三次呼叫outer會彈出遞增的結果。

注意

1 閉包允許內層函數引用父函數中的變量,但是該變數是最終值

/** 
* <body> 
* <ul> 
* <li>one</li> 
* <li>two</li> 
* <li>three</li> 
* <li>one</li> 
* </ul> 
*/ 
var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
lists[ i ].onmouseover = function(){ 
alert(i); 
}; 
}

你会发现当鼠标移过每一个25edfb22a4f469ecb59f1190150159c6元素时,总是弹出4,而不是我们期待的元素下标。这是为什么呢?注意事项里已经讲了(最终值)。显然这种解释过于简单,当mouseover事件调用监听函数时,首先在匿名函数( function(){ alert(i); })内部查找是否定义了 i,结果是没有定义;因此它会向上查找,查找结果是已经定义了,并且i的值是4(循环后的i值);所以,最终每次弹出的都是4。

解决方法一:

var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
(function(index){ 
lists[ index ].onmouseover = function(){ 
alert(index); 
}; 
})(i); 
}

解决方法二:

var lists = document.getElementsByTagName('li'); 
for(var i = 0, len = lists.length; i < len; i++){ 
lists[ i ].$$index = i; //通过在Dom元素上绑定$$index属性记录下标 
lists[ i ].onmouseover = function(){ 
alert(this.$$index); 
}; 
}

解决方法三:

function eventListener(list, index){ 
list.onmouseover = function(){ 
alert(index); 
}; 
} 
var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
eventListener(lists[ i ] , i); 
}

2 内存泄露

使用闭包十分容易造成浏览器的内存泄露,严重情况下会是浏览器挂死

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn