小編給大家分享一下JavaScript如何立即調(diào)用函數(shù)表達(dá)式,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
成都創(chuàng)新互聯(lián)公司致力于互聯(lián)網(wǎng)品牌建設(shè)與網(wǎng)絡(luò)營銷,包括做網(wǎng)站、成都做網(wǎng)站、SEO優(yōu)化、網(wǎng)絡(luò)推廣、整站優(yōu)化營銷策劃推廣、電子商務(wù)、移動互聯(lián)網(wǎng)營銷等。成都創(chuàng)新互聯(lián)公司為不同類型的客戶提供良好的互聯(lián)網(wǎng)應(yīng)用定制及解決方案,成都創(chuàng)新互聯(lián)公司核心團(tuán)隊(duì)10年專注互聯(lián)網(wǎng)開發(fā),積累了豐富的網(wǎng)站經(jīng)驗(yàn),為廣大企業(yè)客戶提供一站式企業(yè)網(wǎng)站建設(shè)服務(wù),在網(wǎng)站建設(shè)行業(yè)內(nèi)樹立了良好口碑。
立即調(diào)用函數(shù)
一、了解立即調(diào)用函數(shù)表達(dá)式
在詳細(xì)了解這個之前,我們來談了解一下“自執(zhí)行”這個叫法,本文對這個功能的叫法也不一定完全對,每個人對他的理解都不一樣,我們在這里用立即調(diào)用
~
立即調(diào)用:
顧名思義,該表達(dá)式一被創(chuàng)建就立即執(zhí)行
。
是一個在定義時就會立即執(zhí)行的 JavaScript 函數(shù)
。
(function (x) { console.log('x + x = ', x + x);})(5) // x + x = 10
這是一個被稱為 自執(zhí)行匿名函數(shù)
的設(shè)計(jì)模式,主要包含兩部分:
第一部分是包圍在 圓括號運(yùn)算符 ()
里的一個匿名函數(shù),這個匿名函數(shù)擁有獨(dú)立的詞法作用域。這不僅避免了外界訪問此 IIFE 中的變量,而且又不會污染全局作用域。
第二部分再一次使用 ()
創(chuàng)建了一個立即執(zhí)行函數(shù)表達(dá)式,JavaScript 引擎到此將直接執(zhí)行函數(shù)。
當(dāng)你聲明一個函數(shù)的時候,通過在后面加括號就可以實(shí)現(xiàn)立即執(zhí)行嗎?
var foo = function(){ console.log('余光');}(); // 余光 成功了! // ...是不是意味著后面加個括弧都可以自動執(zhí)行?function(){ console.log(''余光);}(); // Uncaught SyntaxError: Function statements require a function name// 什么?還需要一個函數(shù)名?不是叫 自執(zhí)行匿名函數(shù)嗎?// 我加上了函數(shù)名function foo(){ console.log('余光');}(); // Uncaught SyntaxError: Unexpected token ')'
很顯然,例子中的第二條和第三條確實(shí)報錯了,而且報錯內(nèi)容不一樣,那么問題出現(xiàn)在哪呢?
二、立即調(diào)用函數(shù)表達(dá)式報錯了?
有時,我們定義函數(shù)之后,立即調(diào)用該函數(shù),這時不能在函數(shù)的定義后面直接加圓括號,這會產(chǎn)生語法錯誤。產(chǎn)生語法錯誤的原因是,function
這個關(guān)鍵字,既可以當(dāng)做語句,也可以當(dāng)做表達(dá)式,比如下邊:
//語句function fn() {};//表達(dá)式var fn = function (){};
為了避免解析上的歧義,JS引擎規(guī)定,如果function出現(xiàn)在行首,一律解析成語句。因此JS引擎看到行首是function關(guān)鍵字以后,認(rèn)為這一段都是函數(shù)定義,不應(yīng)該以括號結(jié)尾
,在它看來括號
只是分組操作符。
// 下面這個function在語法上是沒問題的,但是依然只是一個語句// 加上括號()以后依然會報錯,因?yàn)榉纸M操作符需要包含表達(dá)式 function foo(){ /* code */ }(); // SyntaxError: Unexpected token ) // 但是如果你在括弧()里傳入一個表達(dá)式,將不會有異常拋出// 但是foo函數(shù)依然不會執(zhí)行function foo(){ /* code */ }( 1 ); // 因?yàn)樗耆葍r于下面這個代碼,一個function聲明后面,又聲明了一個毫無關(guān)系的表達(dá)式: function foo(){ /* code */ } ( 1 );
三、使用立即調(diào)用函數(shù)的正確姿勢
要解決上述問題,非常簡單。
我們只需要用大括弧
將代碼的代碼全部括住就行了,因?yàn)镴avaScript里括弧()
里面不能包含語句,所以在這一點(diǎn)上,解析器在解析function關(guān)鍵字的時候,會將相應(yīng)的代碼解析成function表達(dá)式,而不是function聲明。
// 下面2個括弧()都會立即執(zhí)行(function () { /* code */ } ()); // 推薦使用這個(function () { /* code */ })(); // 但是這個也是可以用的
// 由于括弧()和JS的&&,異或,逗號等操作符是在函數(shù)表達(dá)式和函數(shù)聲明上消除歧義的// 所以一旦解析器知道其中一個已經(jīng)是表達(dá)式了,其它的也都默認(rèn)為表達(dá)式了var i = function() { console.log('余光')}(); // 余光true && function() { console.log('余光')}(); // 余光0, function() { console.log('余光') }(); // 余光
// 如果你不在意返回值,或者不怕難以閱讀// 你甚至可以在function前面加一元操作符號//轉(zhuǎn)boolvar res1 = !function () { console.log('余光');}()console.log('res1:', res1); // 余光 true// 轉(zhuǎn)數(shù)字var res2 = +function () { console.log('余光');}()console.log('res2:', res2); // 余光 NaN// 按位非var res3 = ~function () { console.log('余光');}()console.log('res3:', res3); // 余光 NaN
還有一個情況,使用new和void關(guān)鍵字,不過不太常見罷了。
void function() { console.log('余光');}();new function() { console.log('余光');}();
四、常見使用場景
IIFE最常見的功能,就是隔離作用域,在ES6之前JS原生也沒有塊級作用域的概念,所以需要函數(shù)作用域來模擬。
舉例:
var currentTime = (function () { var time = new Date(); var year = time.getFullYear() var month = time.getMonth()+1; var date = time.getDate(); var hour = time.getHours(); var min = time.getMinutes(); return year + '-' + month + '-' + date + ' ' + hour + ':' + min;})()
你仍然可以在其他地方聲明同名變量~
DOM事件添加中,為了兼容現(xiàn)代瀏覽器和IE瀏覽器,我們需要對瀏覽器環(huán)境進(jìn)行一次判斷:
var addEvent = (function(){ if(window.addEventListener) { return function(type, el, fn) { el.addEventListener(type, fn, false); } } else if(window.attachEvent) { return function(type, el, fn) { el.attachEvent('on' + type, fn); } }})();
這里我僅舉個例子,為我的下一篇文章——《JavaScript中的閉包》賣個關(guān)子
var elems = document.getElementsByTagName('a');for (var i = 0; i < elems.length; i++) { (function (lockedInIndex) { elems[i].addEventListener('click', function (e) { e.preventDefault(); alert('I am link #' + lockedInIndex); }, 'false'); })(i);}
當(dāng)函數(shù)變成立即執(zhí)行的函數(shù)表達(dá)式時,表達(dá)式中的變量不能從外部訪問。
(function () { var name = "Barry";})();// 無法從外部訪問變量 namename // 拋出錯誤:"Uncaught ReferenceError: name is not defined"
將 IIFE 分配給一個變量,不是存儲 IIFE 本身,而是存儲 IIFE 執(zhí)行后返回的結(jié)果。
var result = (function () { var name = "Barry"; return name; })(); // IIFE 執(zhí)行后返回的結(jié)果:result; // "Barry"
以上是“JavaScript如何立即調(diào)用函數(shù)表達(dá)式”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!