中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

JavaScript如何立即調用函數表達式

發布時間:2021-03-22 10:39:54 來源:億速云 閱讀:315 作者:小新 欄目:web開發

小編給大家分享一下JavaScript如何立即調用函數表達式,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

立即調用函數

一、了解立即調用函數表達式

1.1 思維導圖

JavaScript如何立即調用函數表達式

1.2 什么是立即調用?

在詳細了解這個之前,我們來談了解一下“自執行”這個叫法,本文對這個功能的叫法也不一定完全對,每個人對他的理解都不一樣,我們在這里用立即調用

立即調用:

  • 顧名思義,該表達式一被創建就立即執行

  • 是一個在定義時就會立即執行的 JavaScript 函數

(function (x) {
    console.log('x + x = ', x + x);})(5) // x + x = 10

這是一個被稱為 自執行匿名函數 的設計模式,主要包含兩部分:

  1. 第一部分是包圍在 圓括號運算符 () 里的一個匿名函數,這個匿名函數擁有獨立的詞法作用域。這不僅避免了外界訪問此 IIFE 中的變量,而且又不會污染全局作用域。

  2. 第二部分再一次使用 () 創建了一個立即執行函數表達式,JavaScript 引擎到此將直接執行函數。

1.3 核心問題

當你聲明一個函數的時候,通過在后面加括號就可以實現立即執行嗎?

var foo = function(){ 
	console.log('余光');}(); // 余光 成功了!
 // ...是不是意味著后面加個括弧都可以自動執行?function(){
	console.log(''余光);}(); // Uncaught SyntaxError: Function statements require a function name// 什么?還需要一個函數名?不是叫 自執行匿名函數嗎?// 我加上了函數名function foo(){
	console.log('余光');}(); // Uncaught SyntaxError: Unexpected token ')'

很顯然,例子中的第二條和第三條確實報錯了,而且報錯內容不一樣,那么問題出現在哪呢?

二、立即調用函數表達式報錯了?

有時,我們定義函數之后,立即調用該函數,這時不能在函數的定義后面直接加圓括號,這會產生語法錯誤。產生語法錯誤的原因是,function 這個關鍵字,既可以當做語句,也可以當做表達式,比如下邊:

//語句function fn() {};//表達式var fn = function (){};

為了避免解析上的歧義,JS引擎規定,如果function出現在行首,一律解析成語句。因此JS引擎看到行首是function關鍵字以后,認為這一段都是函數定義,不應該以括號結尾,在它看來括號只是分組操作符。

// 下面這個function在語法上是沒問題的,但是依然只是一個語句// 加上括號()以后依然會報錯,因為分組操作符需要包含表達式
 function foo(){ /* code */ }(); // SyntaxError: Unexpected token )
 // 但是如果你在括弧()里傳入一個表達式,將不會有異常拋出// 但是foo函數依然不會執行function foo(){ /* code */ }( 1 );
 // 因為它完全等價于下面這個代碼,一個function聲明后面,又聲明了一個毫無關系的表達式: function foo(){ /* code */ }
 ( 1 );

三、使用立即調用函數的正確姿勢

要解決上述問題,非常簡單。

我們只需要用大括弧將代碼的代碼全部括住就行了,因為JavaScript里括弧()里面不能包含語句,所以在這一點上,解析器在解析function關鍵字的時候,會將相應的代碼解析成function表達式,而不是function聲明。

3.1 常見使用姿勢
// 下面2個括弧()都會立即執行(function () { /* code */ } ()); // 推薦使用這個(function () { /* code */ })(); // 但是這個也是可以用的
3.2 不常見的使用姿勢(一)
// 由于括弧()和JS的&&,異或,逗號等操作符是在函數表達式和函數聲明上消除歧義的// 所以一旦解析器知道其中一個已經是表達式了,其它的也都默認為表達式了var i = function() {
    console.log('余光')}(); // 余光true && function() {
    console.log('余光')}(); // 余光0, function() { console.log('余光') }(); // 余光
3.3 不常見的使用姿勢(二)
// 如果你不在意返回值,或者不怕難以閱讀// 你甚至可以在function前面加一元操作符號//轉boolvar res1 = !function () {
    console.log('余光');}()console.log('res1:', res1); // 余光 true// 轉數字var res2 = +function () {
    console.log('余光');}()console.log('res2:', res2); // 余光 NaN// 按位非var res3 = ~function () {
    console.log('余光');}()console.log('res3:', res3); // 余光 NaN
3.4 不常見的使用姿勢(三)

還有一個情況,使用new和void關鍵字,不過不太常見罷了。

void function() {
    console.log('余光');}();new function() {
    console.log('余光');}();

四、常見使用場景

4.1 隔離作用域

IIFE最常見的功能,就是隔離作用域,在ES6之前JS原生也沒有塊級作用域的概念,所以需要函數作用域來模擬。

舉例:

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;})()

你仍然可以在其他地方聲明同名變量~

4.2 惰性函數

DOM事件添加中,為了兼容現代瀏覽器和IE瀏覽器,我們需要對瀏覽器環境進行一次判斷:

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);
        }
    }})();
4.3 用閉包保存狀態

這里我僅舉個例子,為我的下一篇文章——《JavaScript中的閉包》賣個關子

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);}
注意

當函數變成立即執行的函數表達式時,表達式中的變量不能從外部訪問。

(function () { 
    var name = "Barry";})();// 無法從外部訪問變量 namename // 拋出錯誤:"Uncaught ReferenceError: name is not defined"

將 IIFE 分配給一個變量,不是存儲 IIFE 本身,而是存儲 IIFE 執行后返回的結果。

var result = (function () { 
    var name = "Barry"; 
    return name; })(); // IIFE 執行后返回的結果:result; // "Barry"

以上是“JavaScript如何立即調用函數表達式”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

辽阳市| 西安市| 彭州市| 区。| 平乐县| 日喀则市| 安仁县| 乌兰察布市| 教育| 礼泉县| 辉南县| 盐边县| 古田县| 黎平县| 石林| 濮阳县| 资源县| 卢氏县| 双桥区| 富顺县| 阿坝| 太保市| 新巴尔虎左旗| 剑川县| 吉木乃县| 山阴县| 邯郸县| 安新县| 班玛县| 乐山市| 长寿区| 肇源县| 扎兰屯市| 栾城县| 富民县| 龙游县| 延庆县| 潢川县| 汽车| 滁州市| 砚山县|