結(jié)合函數(shù)fn分析:var聲明變量和直接用標(biāo)識(shí)符初始化的細(xì)節(jié)
1.var聲明會(huì)指定變量的作用域访诱,即用 var 操作符定義的變量將成為定義該變量的作用域中的局部變量
--fn執(zhí)行完畢拒贱,在全局中訪問(wèn)c會(huì)報(bào)錯(cuò)說(shuō)變量未定義铐刘,說(shuō)明c的作用域只是在fn函數(shù)里面,而直接使用d = 4聲明初始化的變量則
默認(rèn)是全局作用域镰惦,從c的輸出可以驗(yàn)證
2.直接標(biāo)識(shí)符初始化變量則默認(rèn)是widow的一個(gè)屬性迷守,在全局中可以訪問(wèn),
--《JavaScript面向?qū)ο缶幊讨改稀?第二版)中提到直接標(biāo)識(shí)符初始化的變量是window的一個(gè)屬性,這可以在window.b輸出
為2得到印證旺入。瀏覽器在訪問(wèn)window對(duì)象的屬性和方法可以省略‘window.’,也就是說(shuō)在代碼執(zhí)行時(shí)候遇到?jīng)]有var聲明的標(biāo)示符
會(huì)默認(rèn)把它當(dāng)成window的屬性
--如此說(shuō)來(lái)兑凿,b = 2,這句代碼其實(shí)是為window對(duì)象聲明一個(gè)屬性并初始化, var 聲明的變量不是window的屬性window.a輸
出undefiend是因?yàn)樵谠L問(wèn)window.a的時(shí)候茵瘾,由于在window對(duì)象及其原型鏈中沒(méi)有搜索到標(biāo)示符為a的屬性礼华,所以就為window
聲明了a屬性,但是并沒(méi)有初始化拗秘,有聲明沒(méi)初始化的變量被默認(rèn)賦值為undefined,這就是a的值為undefined而不引起瀏覽器
報(bào)錯(cuò)
--發(fā)現(xiàn)《JavaScript高級(jí)程序設(shè)計(jì)》(第三版)中提到的‘所有在全局作用域中聲明的變量圣絮,函數(shù)都會(huì)成為window對(duì)象的屬性
和方法’這句話不太正確,window屬性必須通過(guò)window.obj或者不使用var直接obj = {}方式定義的變量才會(huì)是window的屬
性;在全局作用域中使用var聲明的變量不是window的屬性雕旨,比如以下的a;
--全局變量的定義可以通過(guò)b = 1,window.a的方式定義扮匠,局部變量通過(guò)var定義捧请,var其實(shí)能夠定義全局變量只是需要在全局環(huán)
境中定義,個(gè)人覺(jué)得它本質(zhì)上還是局部變量,只不過(guò)是var在指定變量的作用域時(shí)候棒搜,這個(gè)作用域剛好是全局作用域疹蛉,作用域存在
它就不會(huì)被釋放,而全局作用域又是依賴全局執(zhí)行環(huán)境存在帮非,全局執(zhí)行環(huán)境只有在頁(yè)面關(guān)閉時(shí)候才會(huì)退出氧吐,使得它能夠在頁(yè)面的每
個(gè)地方訪問(wèn)到。
var a = 1;
b = 2;
function fn(){
var c = 3;
d = 4;
}
fn();
console.log(a); // 1
console.log(c); // Uncaught ReferenceError: c is not defined
console.log(d); // 4
console.log(window.a); // undefined
console.log(window.b); // 2
3.筆者遺留的問(wèn)題末盔,既然b是window的屬性筑舅,那么它?在棧中沒(méi)有分配存儲(chǔ)空間而var聲明的在棧中分配了空間?(至今苦于沒(méi)有學(xué)習(xí)到JavaScript中能夠獲取內(nèi)存信息的方法所以一直沒(méi)有驗(yàn)證陨舱,希望能解決的朋友能夠告訴我一聲)翠拣。
4.以下是結(jié)合以上結(jié)論個(gè)人對(duì) x,window.x游盲,var x 的理解:
1.var 關(guān)鍵字其實(shí)是用來(lái)為變量申請(qǐng)內(nèi)存的误墓,瀏覽器在解析遇到var關(guān)鍵字時(shí)候會(huì)為它聲明的變量在棧中分配存儲(chǔ)空間,那么遇
到 b = 1; 這類不帶var表達(dá)式時(shí)會(huì)認(rèn)為b已經(jīng)分配內(nèi)存了益缎,所以并沒(méi)有為它分配內(nèi)存谜慌;
2.瀏覽器執(zhí)行行流到達(dá) b = 1;時(shí),語(yǔ)句才真正被執(zhí)行賦值操作莺奔,然而我們知道js代碼執(zhí)行中如果曾經(jīng)沒(méi)有被var聲明過(guò)欣范,它會(huì)
當(dāng)成window的一個(gè)屬性,然而window是一個(gè)對(duì)象令哟,window標(biāo)識(shí)符的值是一個(gè)地址存在棧中指向堆內(nèi)存中一個(gè)對(duì)象恼琼,堆內(nèi)存又
是動(dòng)態(tài)分配的,b = 1;對(duì)還沒(méi)分配內(nèi)存的變量b賦值會(huì)為它在堆中分配內(nèi)存(瀏覽器解析時(shí)候已經(jīng)將需要分棧內(nèi)存的變量分配好
內(nèi)存屏富,執(zhí)行時(shí)候除了執(zhí)行環(huán)境變化才會(huì)操作棧內(nèi)存晴竞,這也是為什么棧中保存的數(shù)據(jù)是固定大小的),而f;是不會(huì)分配內(nèi)存的狠半。所
以f沒(méi)被映射到內(nèi)存中噩死。
--以下輸出語(yǔ)句是實(shí)現(xiàn)獲取f的值在打印在控制臺(tái)的功能,計(jì)算機(jī)無(wú)法通過(guò)b對(duì)應(yīng)的地址取到值所以報(bào)錯(cuò)典予,然而對(duì)比一下第二句代
碼console.log(window.f);輸出undefined甜滨,然而不是說(shuō)瀏覽器遇到?jīng)]有var聲明的f會(huì)當(dāng)成瀏覽器屬性嗎?是的這并不矛
盾瘤袖,f和window.f還是不一樣的衣摩,window.f執(zhí)行過(guò)程:a).在window對(duì)象中獲取f的值,如果沒(méi)有找到執(zhí)行下一個(gè)過(guò)程;b).設(shè)
置f的值艾扮,沒(méi)有指定值既琴,默認(rèn)賦值為undefined,而f;語(yǔ)句并不會(huì)調(diào)用設(shè)值方法泡嘴,所以錯(cuò)誤了
3.總結(jié):變量和對(duì)象有沒(méi)有定義是指它是否被分配內(nèi)存甫恩,js中 var和賦值操作發(fā)生時(shí)候,如果操作的變量事先沒(méi)有定義過(guò)酌予,那
么將會(huì)為該變量分配內(nèi)存空間;
?// f標(biāo)識(shí)符是沒(méi)有被聲明過(guò)磺箕,也不存在window對(duì)象中
console.log(f); // 報(bào)錯(cuò):f未定義,
console.log(window.f); // undefined
最后編輯于 :2017.12.04 23:14:05
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者