ES6之塊級作用域

来源:http://www.cnblogs.com/giggle/archive/2016/06/13/5572006.html
-Advertisement-
Play Games

ES6中提供了塊級作用域,分別是let和const,該隨筆分別介紹了它們 ...


一、前言

在ECMAScript6(以下簡稱ES6)之前,ECMAScript的作用域只有兩種:

  1、  全局作用域;

  2、  函數作用域。

正是因為有這兩種作用域,所以在JavaScript中出現一術語--“變數提升(hoisting)”。

如下:

function func(){
    console.log(test);
    var test = 1;
};
func();

在node環境執行上述代碼,結果為:

 

之所以為’undefined’,原因就在於‘變數提升’,在進入func函數時,將所有通過var聲明的變數置前並賦予undefined的值。

但,ES6的到來,為我們提供了‘塊級作用域’。且‘塊級作用域’並不影響var聲明的變數。

What?‘塊級作用域’又不影響var聲明的變數?!!

是的,var聲明的變數的性質和原來一樣,還是具有‘變數提升’的特性。而‘塊級作用域’通過新增命令let和const來體現。

下麵,我們透過新增的let和const命令,協同感受下ES6的塊級作用域。

註:由於let和const屬於ES6,所以都必須使用嚴格模式,否則會報錯。

如下:

let test;

在node環境下,執行代碼:

二、let命令

什麼是let呢?

let和var差不多,都是用來聲明變數的。區別就在於:

  1、  let聲明的變數只在所處於的塊級有效;

  2、  let沒有‘變數提升’的特性,而是‘暫時性死區(temporal dead zone)’特性。

下麵將一一講解。

1、let聲明的變數只在塊級有效。

如下:

'use strict';
function func(args){
    if(true){
        //let聲明i
        let i = 6;
        //在if內列印i值
        console.log('inside: ' + i);
    }
    //在if外,再次列印i值
    console.log('outside: ' + i);
};
func();

在node環境中執行上述代碼,結果如下:

通過demo,我們可以清楚的看見,在第二次(if外)列印i值時,是報錯的。

這因為let聲明的變數i是屬於if內的塊級作用域;而不是像var一樣。

2、let沒有‘變數提升’的特性,而卻有‘暫時性死區(temporal dead zone)’的特性。

如下:

'use strict';
function func(){
    //在let聲明前,列印i
    console.log(i);
    let i;
};
func();

在node環境下執行上述代碼,結果如下:

在let聲明變數前,使用該變數,它是會報錯的,而不是像var那樣會‘變數提升’。

其實說let沒有‘變數提升’的特性,不太對。或者說它提升了,但是ES6規定了在let聲明變數前不能使用該變數。

如下:

'use strict';
var test = 1;
function func(){
    //列印test的值
    console.log(test);
    let test = 2;
};
func();

在node環境下執行上述代碼,結果如下:

如果let聲明的變數沒有變數提升,應該列印’1’(func函數外的test);而他卻報錯,說明它是提升了的,只是規定了不能在其聲明之前使用而已。我們稱這特性叫“暫時性死區(temporal dead zone)”。且這一特性,僅對遵循‘塊級作用域’的命令有效(let、const)。

關於let,最後再通過一個經典案例,體驗下。

如下:

var arr = [];
for(var i = 0; i < 2; i++){
    arr[i] = function(){
        console.log(i);
    };
};
arr[1]();

arr[1]()會輸出2,原因是var聲明的變數會變數提升,且當執行arr[1]函數時,i取自於父函數的i,而此時i已經變為2了,所以就會列印2咯。

以前的常用做法是,利用閉包特性。如下:

var arr = [];
for(var i = 0; i < 2; i++){
    arr[i] = (function(i){
        return function(){
            console.log(i);
        };
    }(i));
};
arr[1]();

又或者屬性方式:

var arr = [];
for(var i = 0; i < 2; i++){
    (arr[i] = function self(){
        console.log(self.x);
    }).x = i;
};
arr[1]();

現在有了let,它聲明的變數作用域為塊級,所以,我們也可以利用let來達到同樣的效果。

如下:

'use strict';
var arr = [];
for(let i = 0; i < 2; i++){
    arr[i] = function(){
        console.log(i);
    };
};
arr[1]();

在node環境下,執行上述代碼結果如下:

 

三、const命令

const命令與let命令一樣,聲明的變數,其作用域都是塊級。

所以const遵循的規則與let相差無二,只是,const是用來聲明恆定變數的。

且,用const聲明恆定變數,聲明的同時就必須賦值,否則會報錯。

如下:

'use strict';
function func(){
    const PI;
    PI = 3.14;
    console.log(PI);
};
func();

在node環境下,執行上述代碼結果如下:

正確的方式為,聲明就得賦值。

如:

const PI = 3.14

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前幾天,舍友去某互聯網公司面前端研發工程師。回來後,他就跟我們聊了下麵試官給他出的題。其中,有一道題是“如何實現iframe高度的自適應?”。好吧,我承認,我聽到iframe這個詞的第一反應就是:這個東西性能差、搜索引擎不友好等等。由於這樣的偏見,還真沒有好好研究一下iframe。其實,iframe ...
  • every():對數組每一項都遍歷,然後每一項都符合要求的話則返回true,否則就返回false some():對數組每一項都遍歷,其中有一項符合要求則返回true,否則返回false filter():對數組的每一項都遍歷,返回其中要求的元素組成的數組 map():對數組每一項都遍歷並且運行給定的 ...
  • 之前的項目一直採用grunt來構建,然後用requirejs做模塊化,requirejs官方有提供grunt的插件來做壓縮合併。現在的項目切到了gulp,模塊化用起了seajs,自然而然地也想到了模塊合併壓縮的問題。然後一開始在解決這個問題的時候,並不是很順利,在npm上並沒有那種特別流行的專門用來... ...
  • 將近20年前,Javascript誕生的時候,只是一種簡單的網頁腳本語言。如果你忘了填寫用戶名,它就跳出一個警告。 如今,它變得幾乎無所不能,從前端到後端,有著各種匪夷所思的用途。程式員用它完成越來越龐大的項目。 Javascript代碼的複雜度也直線上升。單個網頁包含10000行Javascrip ...
  • 無論是提示框還是導航欄都能看到如上圖所示的帶有箭頭的框框,這種箭頭可以通過背景圖片或者是css來實現,本文介紹三種通過css實現帶箭頭的提示框。 通過border屬性思路:兩個三角形,通過定位使兩個三角形相差1px作為邊框。 CSS3 transfrom思路:先做一個兩條邊相同顏色的正方形,然後旋轉 ...
  • jQuery 3.0 在6月9日正式發佈了,3.0 也被稱為下一代的 jQuery 。這個版本從14年10月開始,其中發佈過一次beta 版(2016/1/14,)和候選版(2016/05/20)。一路走來,頗為不易。 文章目錄 一、Data淺析 jQuery 3.0 中的 Data 是內部使用的, ...
  • 1.什麼是HTML標記語言? HTML是表示網頁信息的符號標記語言。 2.HTML的標記和他的屬性 3.語法不區分字母大小寫 <HTML>、<Html>、<html>都是定義相同的標記,但是在編寫網頁的使用儘量使用小寫。 4.文檔註釋 註釋一段內容時使用"<!--"開始,以"-->"結束。 例如: ...
  • H5其實就是H4的一個增強版本,我們在利用H5進行網頁的構造會更簡便,標簽語義更簡潔明瞭。首先,我們要理解HTML4,它是HTML的標記+css2+JavaScript的一些基本應用,簡言之,就是API+語法;而H5無非就是在原先的基礎上面提供了一些新的功能。 1、H5的語義標簽 1):html 的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...