變量及作用域

1.基本類型和引用類型的值

ECMAScript 變量可能包含兩種不同的數(shù)據(jù)類型的值:基本類型值和引用類型值详幽∩富叮基本

類型值指的是那些保存在棧內(nèi)存中的簡單數(shù)據(jù)段浸锨,即這種值完全保存在內(nèi)存中的一個位置。

而引用類型值則是指那些保存在堆內(nèi)存中的對象版姑,意思是變量中保存的實際上只是一個指

針柱搜,這個指針指向內(nèi)存中的另一個位置,該位置保存對象剥险。

將一個值賦給變量時聪蘸,解析器必須確定這個值是基本類型值,還是引用類型值表制〗∨溃基本類

型值有以下幾種:Undefined、Null么介、Boolean娜遵、Number 和String。這些類型在內(nèi)存中分別占

有固定大小的空間壤短,他們的值保存在椛枘猓空間,我們通過按值來訪問的鸽扁。

PS:在某些語言中蒜绽,字符串以對象的形式來表示,因此被認(rèn)為是引用類型桶现。ECMAScript

放棄這一傳統(tǒng)躲雅。

如果賦值的是引用類型的值,則必須在堆內(nèi)存中為這個值分配空間骡和。由于這種值的大小

不固定相赁,因此不能把它們保存到棧內(nèi)存中。但內(nèi)存地址大小的固定的慰于,因此可以將內(nèi)存地址

保存在棧內(nèi)存中钮科。這樣,當(dāng)查詢引用類型的變量時婆赠,先從棧中讀取內(nèi)存地址绵脯,然后再通過地

址找到堆中的值。對于這種休里,我們把它叫做按引用訪問蛆挫。

圖片發(fā)自簡書App

2.動態(tài)屬性

定義基本類型值和引用類型值的方式是相似的:創(chuàng)建一個變量并為該變量賦值。但是妙黍,

當(dāng)這個值保存到變量中以后悴侵,對不同類型值可以執(zhí)行的操作則大相徑庭。

var box = new Object(); //創(chuàng)建引用類型

box.name = 'Lee'; //新增一個屬性

alert(box.name); //輸出

如果是基本類型的值添加屬性的話拭嫁,就會出現(xiàn)問題了可免。

var box = 'Lee'; //創(chuàng)建一個基本類型

box.age = 27; //給基本類型添加屬性

alert(box.age); //undefined

3.復(fù)制變量值

在變量復(fù)制方面抓于,基本類型和引用類型也有所不同〗浇瑁基本類型復(fù)制的是值本身捉撮,而引用

類型復(fù)制的是地址。

var box = 'Lee'; //在棧內(nèi)存生成一個box 'Lee'

var box2 = box; //在棧內(nèi)存再生成一個box2 'Lee'

圖片發(fā)自簡書App

box2 是雖然是box1 的一個副本逮刨,但從圖示可以看出呕缭,它是完全獨立的。也就是說修己,兩

個變量分別操作時互不影響恢总。

var box = new Object(); //創(chuàng)建一個引用類型

box.name = 'Lee'; //新增一個屬性

var box2 = box;

圖片發(fā)自簡書App

在引用類型中,box2 其實就是box睬愤,因為他們指向的是同一個對象片仿。如果這個對象中的

name 屬性被修改了,box2.name 和box.name 輸出的值都會被相應(yīng)修改掉了尤辱。

4.傳遞參數(shù)

ECMAScript 中所有函數(shù)的參數(shù)都是按值傳遞的砂豌,言下之意就是說,參數(shù)不會按引用傳

遞光督,雖然變量有基本類型和引用類型之分阳距。

function box(num) { //按值傳遞,傳遞的參數(shù)是基本類型

num += 10; //這里的num 是局部變量结借,全局無效

return num;

}

var num = 50;

var result = box(num);

alert(result); //60

alert(num); //50

PS:以上的代碼中筐摘,傳遞的參數(shù)是一個基本類型的值。而函數(shù)里的num 是一個局部變

量船老,和外面的num 沒有任何聯(lián)系咖熟。

下面給出一個參數(shù)作為引用類型的例子。

function box(obj) { //按值傳遞柳畔,傳遞的參數(shù)是引用類型

obj.name = 'Lee';

}

var p = new Object();

box(p);

alert(p.name);

PS:如果存在按引用傳遞的話馍管,那么函數(shù)里的那個變量將會是全局變量,在外部也可

以訪問薪韩。比如PHP 中确沸,必須在參數(shù)前面加上&符號表示按引用傳遞。而ECMAScript 沒有這

些俘陷,只能是局部變量罗捎。可以在PHP 中了解一下岭洲。

PS:所以按引用傳遞和傳遞引用類型是兩個不同的概念。

function box(obj) {

obj.name = 'Lee';

var obj = new Object(); //函數(shù)內(nèi)部又創(chuàng)建了一個對象

obj.name = 'Mr.'; //并沒有替換掉原來的obj

}

最后得出結(jié)論坎匿,ECMAScript 函數(shù)的參數(shù)都將是局部變量盾剩,也就是說雷激,沒有按引用傳遞。

5.檢測類型

要檢測一個變量的類型告私,我們可以通過typeof 運算符來判別屎暇。諸如:

var box = 'Lee';

alert(typeof box); //string

雖然typeof 運算符在檢查基本數(shù)據(jù)類型的時候非常好用,但檢測引用類型的時候驻粟,它就

不是那么好用了根悼。通常,我們并不想知道它是不是對象蜀撑,而是想知道它到底是什么類型的對

象挤巡。因為數(shù)組也是object,null 也是Object 等等酷麦。

這時我們應(yīng)該采用instanceof 運算符來查看矿卑。

var box = [1,2,3];

alert(box instanceof Array); //是否是數(shù)組

var box2 = {};

alert(box2 instanceof Object); //是否是對象

var box3 = /g/;

alert(box3 instanceof RegExp); //是否是正則表達(dá)式

var box4 = new String('Lee');

alert(box4 instanceof String); //是否是字符串對象

PS:當(dāng)使用instanceof 檢查基本類型的值時,它會返回false沃饶。

5.執(zhí)行環(huán)境及作用域

執(zhí)行環(huán)境是JavaScript 中最為重要的一個概念母廷。執(zhí)行環(huán)境定義了變量或函數(shù)有權(quán)訪問的

其他數(shù)據(jù),決定了它們各自的行為糊肤。

全局執(zhí)行環(huán)境是最外圍的執(zhí)行環(huán)境琴昆。在Web 瀏覽器中,全局執(zhí)行環(huán)境被認(rèn)為是window

對象馆揉。因此所有的全局變量和函數(shù)都是作為window 對象的屬性和方法創(chuàng)建的业舍。

var box = 'blue'; //聲明一個全局變量

function setBox() {

alert(box); //全局變量可以在函數(shù)里訪問

}

setBox(); //執(zhí)行函數(shù)

全局的變量和函數(shù),都是window 對象的屬性和方法把介。

var box = 'blue';

function setBox() {

alert(window.box); //全局變量即window 的屬性

}

window.setBox(); //全局函數(shù)即window 的方法

PS:當(dāng)執(zhí)行環(huán)境中的所有代碼執(zhí)行完畢后勤讽,該環(huán)境被銷毀,保存在其中的所有變量和

函數(shù)定義也隨之銷毀拗踢。如果是全局環(huán)境下脚牍,需要程序執(zhí)行完畢,或者網(wǎng)頁被關(guān)閉才會銷毀巢墅。

PS:每個執(zhí)行環(huán)境都有一個與之關(guān)聯(lián)的變量對象诸狭,就好比全局的window 可以調(diào)用變量

和屬性一樣。局部的環(huán)境也有一個類似window 的變量對象君纫,環(huán)境中定義的所有變量和函數(shù)

都保存在這個對象中驯遇。(我們無法訪問這個變量對象,但解析器會處理數(shù)據(jù)時后臺使用它)

函數(shù)里的局部作用域里的變量替換全局變量蓄髓,但作用域僅限在函數(shù)體內(nèi)這個局部環(huán)境叉庐。

var box = 'blue';

function setBox() {

var box = 'red'; //這里是局部變量,出來就不認(rèn)識了

alert(box);

}

setBox();

alert(box);

通過傳參会喝,可以替換函數(shù)體內(nèi)的局部變量陡叠,但作用域僅限在函數(shù)體內(nèi)這個局部環(huán)境玩郊。

var box = 'blue';

function setBox(box) { //通過傳參,替換了全局變量

alert(box);

}

setBox('red');

alert(box);

函數(shù)體內(nèi)還包含著函數(shù)枉阵,只有這個函數(shù)才可以訪問內(nèi)一層的函數(shù)译红。

var box = 'blue';

function setBox() {

function setColor() {

var b = 'orange';

alert(box);

alert(b);

}

setColor(); //setColor()的執(zhí)行環(huán)境在setBox()內(nèi)

}

setBox();

PS:每個函數(shù)被調(diào)用時都會創(chuàng)建自己的執(zhí)行環(huán)境。當(dāng)執(zhí)行到這個函數(shù)時兴溜,函數(shù)的環(huán)境

就會被推到環(huán)境棧中去執(zhí)行侦厚,而執(zhí)行后又在環(huán)境棧中彈出(退出),把控制權(quán)交給上一級的執(zhí)

行環(huán)境拙徽。

PS:當(dāng)代碼在一個環(huán)境中執(zhí)行時刨沦,就會形成一種叫做作用域鏈的東西。它的用途是保

證對執(zhí)行環(huán)境中有訪問權(quán)限的變量和函數(shù)進(jìn)行有序訪問斋攀。作用域鏈的前端已卷,就是執(zhí)行環(huán)境的

變量對象。

6.沒有塊級作用域

塊級作用域表示諸如if 語句等有花括號封閉的代碼塊淳蔼,所以侧蘸,支持條件判斷來定義變

量。

if (true) { //if 語句代碼塊沒有局部作用域

var box = 'Lee';

}

alert(box);

for 循環(huán)語句也是如此

for (var i = 0; i < 10; i ++) { //沒有局部作用域

var box = 'Lee';

}

alert(i);

alert(box);

var 關(guān)鍵字在函數(shù)里的區(qū)別

function box(num1, num2) {

var sum = num1 + num2; //如果去掉var 就是全局變量了

return sum;

}

alert(box(10,10));

alert(sum); //報錯

PS:非常不建議不使用var 就初始化變量鹉梨,因為這種方法會導(dǎo)致各種意外發(fā)生讳癌。所以初

始化變量的時候一定要加上var。

一般確定變量都是通過搜索來確定該標(biāo)識符實際代表什么存皂。

var box = 'blue';

function getBox() {

return box; //代表全局box

} //如果加上函數(shù)體內(nèi)加上var box = 'red'

alert(getBox()); //那么最后返回值就是red

PS:變量查詢中晌坤,訪問局部變量要比全局變量更快,因為不需要向上搜索作用域鏈旦袋。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末骤菠,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子疤孕,更是在濱河造成了極大的恐慌商乎,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,914評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件祭阀,死亡現(xiàn)場離奇詭異鹉戚,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)专控,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評論 2 383
  • 文/潘曉璐 我一進(jìn)店門抹凳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人伦腐,你說我怎么就攤上這事赢底。” “怎么了?”我有些...
    開封第一講書人閱讀 156,531評論 0 345
  • 文/不壞的土叔 我叫張陵幸冻,是天一觀的道長嗅剖。 經(jīng)常有香客問我,道長嘁扼,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,309評論 1 282
  • 正文 為了忘掉前任黔攒,我火速辦了婚禮趁啸,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘督惰。我一直安慰自己不傅,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,381評論 5 384
  • 文/花漫 我一把揭開白布赏胚。 她就那樣靜靜地躺著访娶,像睡著了一般。 火紅的嫁衣襯著肌膚如雪觉阅。 梳的紋絲不亂的頭發(fā)上崖疤,一...
    開封第一講書人閱讀 49,730評論 1 289
  • 那天,我揣著相機(jī)與錄音典勇,去河邊找鬼劫哼。 笑死,一個胖子當(dāng)著我的面吹牛割笙,可吹牛的內(nèi)容都是我干的权烧。 我是一名探鬼主播,決...
    沈念sama閱讀 38,882評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼伤溉,長吁一口氣:“原來是場噩夢啊……” “哼般码!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起乱顾,我...
    開封第一講書人閱讀 37,643評論 0 266
  • 序言:老撾萬榮一對情侶失蹤板祝,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后糯耍,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體扔字,經(jīng)...
    沈念sama閱讀 44,095評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,448評論 2 325
  • 正文 我和宋清朗相戀三年温技,在試婚紗的時候發(fā)現(xiàn)自己被綠了革为。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,566評論 1 339
  • 序言:一個原本活蹦亂跳的男人離奇死亡舵鳞,死狀恐怖震檩,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤抛虏,帶...
    沈念sama閱讀 34,253評論 4 328
  • 正文 年R本政府宣布博其,位于F島的核電站,受9級特大地震影響迂猴,放射性物質(zhì)發(fā)生泄漏慕淡。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,829評論 3 312
  • 文/蒙蒙 一沸毁、第九天 我趴在偏房一處隱蔽的房頂上張望峰髓。 院中可真熱鬧,春花似錦息尺、人聲如沸携兵。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,715評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽徐紧。三九已至,卻和暖如春炭懊,著一層夾襖步出監(jiān)牢的瞬間并级,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,945評論 1 264
  • 我被黑心中介騙來泰國打工侮腹, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留死遭,地道東北人。 一個月前我還...
    沈念sama閱讀 46,248評論 2 360
  • 正文 我出身青樓凯旋,卻偏偏與公主長得像呀潭,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子至非,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,440評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 謹(jǐn)記 當(dāng)你感覺累的時候荒椭,說明你還活著谐鼎,當(dāng)你突然感覺不到的累那一刻,也就意味著你已經(jīng)被社會淘汰了趣惠,人活著就的受累狸棍,因...
    長風(fēng)留言閱讀 1,291評論 3 7
  • 《ijs》速成開發(fā)手冊3.0 官方用戶交流:iApp開發(fā)交流(1) 239547050iApp開發(fā)交流(2) 10...
    葉染柒丶閱讀 5,086評論 0 7
  • 第5章 引用類型(返回首頁) 本章內(nèi)容 使用對象 創(chuàng)建并操作數(shù)組 理解基本的JavaScript類型 使用基本類型...
    大學(xué)一百閱讀 3,216評論 0 4
  • 第一章: JS簡介 從當(dāng)初簡單的語言,變成了現(xiàn)在能夠處理復(fù)雜計算和交互味悄,擁有閉包草戈、匿名函數(shù), 甚至元編程等...
    LaBaby_閱讀 1,651評論 0 6
  • ECMAScript有兩種開發(fā)模式:1.函數(shù)式(過程化)侍瑟,2.面向?qū)ο?OOP)唐片。面向?qū)ο蟮恼Z言有一個標(biāo)志丙猬,那就是...
    lovelydong閱讀 640評論 0 2