一、let和var區(qū)別
1.關(guān)于變量提升,var能變量提升,let不能
// 關(guān)于var 如下所示console.log(a); //輸出undefined,此時(shí)就是變量提升var a = 2; console.log(a); //2//相當(dāng)于下面的代碼var a; //聲明且初始化為undefinedconsole.log(a); //輸出undefineda=2; //賦值console.log(a); //2// 關(guān)于let 如下所示console.log(a); // 報(bào)錯(cuò)ReferenceErrorlet a = 2;//相當(dāng)于在第一行先聲明a但沒有初始化,直到賦值時(shí)才初始化//直接用let聲明變量不賦值是會(huì)打印undefined,這時(shí)候初始化了let a;console.log(a);//值為undefined
2.暫時(shí)性死區(qū):塊級(jí)作用域內(nèi)存在let命令,它所聲明的變量就“綁定”這個(gè)區(qū)域,不再受外部的影響重點(diǎn)內(nèi)容,簡(jiǎn)而言之,就是某個(gè)代碼塊有l(wèi)et指令,即使外部有名稱相同的變量,該代碼塊的同名變量與外部的變量也互不干擾。而var不會(huì),如下所示:
//letvar a = 123;if (true) { let a="abc"; console.log(a); //輸出abc }console.log(a); //輸出值為123,全局a與局部a互不影響//varvar a = 123;if (true) { var a="abc"; console.log(a); //輸出abc }console.log(a); //輸出值為abc,全局的已被改變
總之,在代碼塊內(nèi),使用let命令聲明變量之前,該變量都是不可用的。這在語(yǔ)法上,稱為“暫時(shí)性死區(qū)”(temporal dead zone,簡(jiǎn)稱 TDZ)。例子如下:
var tmp=1;if (true) { // TDZ開始 tmp = 'abc'; // ReferenceError console.log(tmp); // ReferenceError let tmp; // TDZ結(jié)束 console.log(tmp); // undefined tmp = 123; console.log(tmp); // 123}console.log(tmp); //
3.let聲明綁定的代碼塊內(nèi),不能重復(fù)聲明同一個(gè)變量,var可以
//a不能重復(fù)聲明function sub() { let a = 10; var a = 1;} //報(bào)錯(cuò),Identifier 'a' has already been declaredfunction sub() { let a = 10; let a = 1;} //同上function sub() { let a = 10; {let a = 1;} //此時(shí)不在同一個(gè)代碼塊,不會(huì)報(bào)錯(cuò)} //var可以重復(fù)聲明,不會(huì)報(bào)錯(cuò)function sub() { var a = 10; var a = 1;}
4.類似for循環(huán)的代碼塊,let只在代碼塊內(nèi)部有效,var在代碼塊外部也有效
//let只在代碼塊內(nèi)部有效for (let i = 0; i < 10; i++) {}console.log(i); //報(bào)錯(cuò)ReferenceError: i is not defined//var在代碼塊外部也有效for (let i = 0; i < 10; i++) {}console.log(i); //101let在for循環(huán)內(nèi)特別之處:就是設(shè)置循環(huán)變量的那部分是一個(gè)父作用域,而循環(huán)體內(nèi)部是一個(gè)單獨(dú)的子作用域。//只在父作用域var a = [];for (let i = 0; i < 10; i++) { a[i] = function () { console.log(i); };}a[6](); // 6//子作用域重新聲明var a = [];for (let i = 0; i < 10; i++) { a[i] = function () { let i=3; //重新賦值 console.log(i); };}a[6](); // 3 ,取得新的值
二、let和const
1、相同點(diǎn):
A、變量不提升。
B、暫時(shí)性死區(qū),只能在聲明的位置后面使用。
C、不可重復(fù)聲明。
2、不同點(diǎn):
let聲明的變量可以改變。
const聲明一個(gè)只讀的常量。一旦聲明,常量的值就不能改變,且聲明的時(shí)候必須初始化賦值。
let a; //undefined
const b;//報(bào)錯(cuò),聲明的時(shí)候必須賦值
let a=1;
a=2; //可改變
const b=1;
b=2; //報(bào)錯(cuò),不能改變值
//一些自己覺得要注意的點(diǎn)
let a=null; //a=null
a=undefined; //a=undefined
a=2; //a=2
const a=null; //a=null,const也可以定義null和undefined
const b=undefined; //b=undefined
b=2; //報(bào)錯(cuò),不能改變值
本質(zhì):
const實(shí)際上保證的,并不是變量的不得改動(dòng),而是變量指向的那個(gè)內(nèi)存地址所保存的數(shù)據(jù)不得改動(dòng)。
A、五種基本數(shù)據(jù)類型(Number,String,Boolean,Undefined,Null):值就保存在變量指向的那個(gè)內(nèi)存地址,等同于常量。不能改變值。
B、復(fù)雜數(shù)據(jù)類型(Object:數(shù)組、對(duì)象):該類型變量名不指向數(shù)據(jù),而是指向數(shù)據(jù)所在的地址,const只保證變量名指向的地址不變,并不保證改地址的數(shù)據(jù)不變,因此可以對(duì)該地址的屬性值進(jìn)行修改,但是不能改變地址指向。
const a=[];a.push("Hello"); //可執(zhí)行,改地址的屬性值可以修改a.length=0; //可執(zhí)行,同上a=["Tom"]; //報(bào)錯(cuò),不能改變地址指向const b ={};b.prop=123; //為b添加一個(gè)屬性,可以成功b.prop //123b={}; //將b指向另外一個(gè)地址,就會(huì)報(bào)錯(cuò)如果真的想將對(duì)象凍結(jié),應(yīng)該使用Object.freeze方法。const b=Object.freeze({});// 常規(guī)模式時(shí),下面一行不起作用,b.prop為undefined// 嚴(yán)格模式時(shí),該行會(huì)報(bào)錯(cuò)b.prop = 123;
總結(jié)
以上所述是小編給大家介紹的var,let,const的異同點(diǎn),希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)武林網(wǎng)網(wǎng)站的支持!
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注