亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > JS > 正文

JavaScript變量作用域及內存問題實例分析

2024-05-06 16:51:57
字體:
來源:轉載
供稿:網友

本文實例講述了JavaScript變量作用域及內存問題。分享給大家供大家參考,具體如下:

學習要點:

1.變量及作用域
2.內存問題

JavaScript的變量與其他語言的變量有很大區別。JavaScript變量是松散型的(不強制類型)本質,決定了它只是在特定時間用于保存特定值的一個名字而已。由于不存在定義某個變量必須要保存何種數據類型值的規則,變量的值及其數據類型可以在腳本的生命周期內改變。

一.變量及作用域

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

ECMAScript變量可能包含兩種不同的數據類型的值:基本類型值和引用類型值。基本類型值指的是那些保存在棧內存中的簡單數據段,即這種值完全保存在內存中的一個位置。而引用類型值則是指那些保存在堆內存中的對象,意思是變量中保存的實際上只是一個指針,這個指針指向內存中的另一個位置,該位置保存對象。

將一個值賦給變量時,解析器必須確定這個值是基本類型值,還是引用類型值。基本類型值有以下幾種:Undefined、Null、Boolean、NumberString。這些類型在內存中分別占有固定大小的空間,他們的值保存在??臻g,我們通過按值來訪問的。

PS:在某些語言中,字符串以對象的形式來表示,因此被認為是引用類型。ECMAScript放棄這一傳統。

如果賦值的是引用類型的值,則必須在堆內存中為這個值分配空間。由于這種值的大小不固定,因此不能把它們保存到棧內存中。但內存地址大小的固定的,因此可以將內存地址保存在棧內存中。這樣,當查詢引用類型的變量時,先從棧中讀取內存地址,然后再通過地址找到堆中的值。對于這種,我們把它叫做按引用訪問。

2.動態屬性

定義基本類型值和引用類型值的方式是相似的:創建一個變量并為該變量賦值。但是,當這個值保存到變量中以后,對不同類型值可以執行的操作則大相徑庭。

var box = new Object(); //創建引用類型box.name = 'Lee'; //新增一個屬性alert(box.name); //輸出

如果是基本類型的值添加屬性的話,就會出現問題了。

var box = 'Lee'; //創建一個基本類型box.age = 27; //給基本類型添加屬性alert(box.age); //undefined

3.復制變量值

在變量復制方面,基本類型和引用類型也有所不同?;绢愋蛷椭频氖侵当旧恚妙愋蛷椭频氖堑刂?。

var box = 'Lee'; //在棧內存生成一個box 'Lee'var box2 = box; //在棧內存再生成一個box2 'Lee'

box2是雖然是box1的一個副本,但從圖示可以看出,它是完全獨立的。也就是說,兩個變量分別操作時互不影響。

var box = new Object(); //創建一個引用類型box.name = 'Lee'; //新增一個屬性var box2 = box; //把引用地址賦值給box2

在引用類型中,box2其實就是box,因為他們指向的是同一個對象。如果這個對象中的name屬性被修改了,box2.name和box.name輸出的值都會被相應修改掉了。

4.傳遞參數

ECMAScript中所有函數的參數都是按值傳遞的,言下之意就是說,參數不會按引用傳遞,雖然變量有基本類型和引用類型之分。

function box(num) { //按值傳遞,傳遞的參數是基本類型  num += 10; //這里的num是局部變量,全局無效  return num;}var num = 50;var result = box(num);alert(result); //60alert(num); //50

PS:以上的代碼中,傳遞的參數是一個基本類型的值。而函數里的num是一個局部變量,和外面的num沒有任何聯系。

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

function box(obj) { //按值傳遞,傳遞的參數是引用類型  obj.name = 'Lee';}var p = new Object();box(p);alert(p.name);

PS:如果存在按引用傳遞的話,那么函數里的那個變量將會是全局變量,在外部也可以訪問。比如PHP中,必須在參數前面加上&符號表示按引用傳遞。而ECMAScript沒有這些,只能是局部變量??梢栽赑HP中了解一下。

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

function box(obj) {  obj.name = 'Lee';  var obj = new Object(); //函數內部又創建了一個對象  obj.name = 'Mr.'; //并沒有替換掉原來的obj}

最后得出結論,ECMAScript函數的參數都將是局部變量,也就是說,沒有按引用傳遞。

5.檢測類型

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

var box = 'Lee';alert(typeof box); //string

雖然typeof運算符在檢查基本數據類型的時候非常好用,但檢測引用類型的時候,它就不是那么好用了。通常,我們并不想知道它是不是對象,而是想知道它到底是什么類型的對象。因為數組也是object,null也是Object等等。

這時我們應該采用instanceof運算符來查看。

var box = [1,2,3];alert(box instanceof Array); //是否是數組var box2 = {};alert(box2 instanceof Object); //是否是對象var box3 = /g/;alert(box3 instanceof RegExp); //是否是正則表達式var box4 = new String('Lee');alert(box4 instanceof String); //是否是字符串對象

PS:當使用instanceof檢查基本類型的值時,它會返回false。

5.執行環境及作用域

執行環境是JavaScript中最為重要的一個概念。執行環境定義了變量或函數有權訪問的其他數據,決定了它們各自的行為。

全局執行環境是最外圍的執行環境。在Web瀏覽器中,全局執行環境被認為是window對象。因此所有的全局變量和函數都是作為window對象的屬性和方法創建的。

var box = 'blue'; //聲明一個全局變量function setBox() {  alert(box); //全局變量可以在函數里訪問}setBox(); //執行函數

全局的變量和函數,都是window對象的屬性和方法。

var box = 'blue';function setBox() {  alert(window.box); //全局變量即window的屬性}window.setBox(); //全局函數即window的方法

PS:當執行環境中的所有代碼執行完畢后,該環境被銷毀,保存在其中的所有變量和函數定義也隨之銷毀。如果是全局環境下,需要程序執行完畢,或者網頁被關閉才會銷毀。

PS:每個執行環境都有一個與之關聯的變量對象,就好比全局的window可以調用變量和屬性一樣。局部的環境也有一個類似window 的變量對象,環境中定義的所有變量和函數都保存在這個對象中。(我們無法訪問這個變量對象,但解析器會處理數據時后臺使用它)

函數里的局部作用域里的變量替換全局變量,但作用域僅限在函數體內這個局部環境。

var box = 'blue';function setBox() {  var box = 'red'; //這里是局部變量,出來就不認識了  alert(box);}setBox();alert(box);

通過傳參,可以替換函數體內的局部變量,但作用域僅限在函數體內這個局部環境。

var box = 'blue';function setBox(box) { //通過傳參,替換了全局變量  alert(box);}setBox('red');alert(box);

函數體內還包含著函數,只有這個函數才可以訪問內一層的函數。

var box = 'blue';function setBox() {  function setColor() {var b = 'orange';    alert(box);alert(b);  }  setColor(); //setColor()的執行環境在setBox()內}setBox();

PS:每個函數被調用時都會創建自己的執行環境。當執行到這個函數時,函數的環境就會被推到環境棧中去執行,而執行后又在環境棧中彈出(退出),把控制權交給上一級的執行環境。

PS:當代碼在一個環境中執行時,就會形成一種叫做作用域鏈的東西。它的用途是保證對執行環境中有訪問權限的變量和函數進行有序訪問。作用域鏈的前端,就是執行環境的變量對象。

6.沒有塊級作用域

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

if (true) { //if語句代碼塊沒有局部作用域  var box = 'Lee';}alert(box);

for循環語句也是如此

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

var關鍵字在函數里的區別

function box(num1, num2) {  var sum = num1 + num2; //如果去掉var就是全局變量了  return sum;}alert(box(10,10));alert(sum); //報錯

PS:非常不建議不使用var就初始化變量,因為這種方法會導致各種意外發生。所以初始化變量的時候一定要加上var。

一般確定變量都是通過搜索來確定該標識符實際代表什么。

var box = 'blue';function getBox() {  return box; //代表全局box} //如果加上函數體內加上var box = 'red'alert(getBox()); //那么最后返回值就是red

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

二.內存問題

JavaScript具有自動垃圾收集機制,也就是說,執行環境會負責管理代碼執行過程中使用的內存。其他語言比如C和C++,必須手工跟蹤內存使用情況,適時的釋放,否則會造成很多問題。而JavaScript則不需要這樣,它會自行管理內存分配及無用內存的回收。

JavaScript最常用的垃圾收集方式是標記清除。垃圾收集器會在運行的時候給存儲在內存中的變量加上標記。然后,它會去掉環境中正在使用變量的標記,而沒有被去掉標記的變量將被視為準備刪除的變量。最后,垃圾收集器完成內存清理工作,銷毀那些帶標記的值并回收他們所占用的內存空間。

垃圾收集器是周期性運行的,這樣會導致整個程序的性能問題。比如IE7以前的版本,它的垃圾收集器是根據內存分配量運行的,比如256個變量就開始運行垃圾收集器,這樣,就不得不頻繁地運行,從而降低的性能。

一般來說,確保占用最少的內存可以讓頁面獲得更好的性能。那么優化內存的最佳方案,就是一旦數據不再有用,那么將其設置為null來釋放引用,這個做法叫做解除引用。這一做法適用于大多數全局變量和全局對象。

var o = {  name : 'Lee'};o = null; //解除對象引用,等待垃圾收集器回收

希望本文所述對大家JavaScript程序設計有所幫助。


注:相關教程知識閱讀請移步到JavaScript/Ajax教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品高清免费在线观看| 国产精品久久久久久久天堂| 日韩中文字幕精品视频| 欧美中文字幕在线观看| 久久的精品视频| 亚洲欧美精品伊人久久| 久久久久久国产| 亚洲女人被黑人巨大进入| 日韩成人av一区| 欧美亚洲午夜视频在线观看| 国产999精品视频| 国产精品视频地址| 中文字幕日韩精品在线观看| 91精品91久久久久久| 欧美精品激情blacked18| 欧美激情视频在线免费观看 欧美视频免费一| 日韩欧美精品在线观看| 国产精品色悠悠| 国产福利视频一区二区| 欧美激情视频网站| 视频在线观看一区二区| 国产精品久久久久久av福利| 日韩av在线网页| 亚洲视频精品在线| 91美女片黄在线观看游戏| 91视频国产一区| 日韩av在线免播放器| 欧美日韩中文在线观看| 欧美亚洲日本网站| 欧美放荡办公室videos4k| 欧美超级乱淫片喷水| 国产在线精品一区免费香蕉| 日韩欧美精品网站| 久久久久九九九九| 国产98色在线| 久久久国产精品一区| 亚洲精品av在线| 欧美中在线观看| 亚洲欧美精品在线| 九九热在线精品视频| 亚洲国产精品免费| 亚洲精品不卡在线| 亚洲激情中文字幕| 亚洲人成网站在线播| 亚洲一区二区黄| 欧美最猛性xxxx| 国产精品久久久久久一区二区| 国产精品com| 国产成人高潮免费观看精品| 国产欧美精品一区二区三区介绍| 久久精品青青大伊人av| 久久精品久久久久| 九九热这里只有精品免费看| 精品丝袜一区二区三区| 亚洲国产精品大全| 亚洲精品二三区| 欧美日本黄视频| 欧美亚洲一区在线| 亚洲欧美日韩国产成人| 97久久精品人搡人人玩| 国产成人精品视频| 97在线观看免费| 久久久久久久色| 午夜精品免费视频| 日韩精品在线观看网站| www.99久久热国产日韩欧美.com| 国内精品视频一区| 一色桃子一区二区| 日韩精品在线私人| 欧美一级片一区| 久久久av网站| 欧美整片在线观看| 久热99视频在线观看| 亚洲美腿欧美激情另类| 91精品国产综合久久香蕉的用户体验| 九九九久久国产免费| www高清在线视频日韩欧美| 亚洲天堂免费在线| 色老头一区二区三区在线观看| 亚洲精品久久久久中文字幕二区| 国产精品美女在线观看| 欧美天天综合色影久久精品| 久久在线视频在线| 精品一区二区三区四区| 懂色av一区二区三区| 久久精品色欧美aⅴ一区二区| 欧美在线视频在线播放完整版免费观看| 国产三级精品网站| 精品国产户外野外| 日韩av电影手机在线| 久久久久久久久久久免费| 亚洲国产精品久久精品怡红院| 亚洲成人教育av| 亚洲精品福利免费在线观看| 92看片淫黄大片看国产片| 98午夜经典影视| 欧美二区在线播放| 高清视频欧美一级| 九九热精品视频在线播放| 成人黄色免费在线观看| 91精品国产色综合久久不卡98口| 在线观看91久久久久久| 国产一区二区三区在线播放免费观看| 在线观看久久久久久| 夜夜躁日日躁狠狠久久88av| 国产精品久久久久久婷婷天堂| 国产精品成人av在线| 亚洲美女又黄又爽在线观看| 伊人亚洲福利一区二区三区| 国产成人精品电影| 国产精品免费一区二区三区都可以| 国产日韩欧美在线| 欧美另类第一页| 亚洲欧美激情一区| 国产精品一区二区久久| 成人黄色午夜影院| 日韩欧美国产中文字幕| 久久久免费精品| 欧美香蕉大胸在线视频观看| 日韩免费在线免费观看| 欧美黑人一区二区三区| 国产精品影院在线观看| 欧美激情在线观看| 中文字幕日韩专区| 欧美成人国产va精品日本一级| 久久久免费观看视频| 欧美成人亚洲成人日韩成人| 国产视频精品自拍| 九九精品视频在线观看| 亚洲视频在线免费观看| 九九精品在线观看| 欧美在线精品免播放器视频| 亚洲第一区中文99精品| 亚洲午夜性刺激影院| 亚洲va欧美va在线观看| 91久久久久久久久久久久久| 这里精品视频免费| 国产精品伦子伦免费视频| 国内成人精品一区| 欧美性jizz18性欧美| 91九色国产视频| 奇米一区二区三区四区久久| 精品一区二区三区电影| 国产精品夫妻激情| 色综合久久久888| 欧美精品激情在线观看| 久久久久久久久久久网站| 国产ts人妖一区二区三区| 狠狠躁18三区二区一区| 成人欧美一区二区三区在线湿哒哒| 久久精品视频一| 久久成人精品视频| 欧美激情二区三区| 国产精品午夜视频| 色黄久久久久久| 久久99视频精品| 亚洲春色另类小说| 国产精品丝袜一区二区三区| 亚洲欧美激情精品一区二区| 欧美国产日韩中文字幕在线| 欧美体内谢she精2性欧美| 成人久久久久久久| 中文字幕日韩精品在线观看| 久久综合伊人77777尤物|