欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

js變量、作用域及內(nèi)存詳解

 更新時間:2014年09月23日 08:57:52   投稿:hebedich  
本文主要詳細分析了JS變量,作用域以及內(nèi)存問題,同時附上非常多的實例,方便大家理解這3個概念,是篇不可多得的文章,希望對大家有所幫助

基本類型值有:undefined,NUll,Boolean,Number和String,這些類型分別在內(nèi)存中占有固定的大小空間,他們的值保存在??臻g,我們通過按值來訪問的。

(1)值類型:數(shù)值、布爾值、null、undefined。
(2)引用類型:對象、數(shù)組、函數(shù)。

如果賦值的是引用類型的值,則必須在堆內(nèi)存中為這個值分配空間。由于這種值的大小不固定(對象有很多屬性和方法),因此不能把他們保存到棧內(nèi)存中。但內(nèi)存地址大小是固定的,因此可以將內(nèi)存地址保存在棧內(nèi)存中。

<script type="text/javascript”>
var box = new Object(); //創(chuàng)建一個引用類型
var box = "lee";  //基本類型值是字符串
box.age = 23;  //基本類型值添加屬性很怪異,因為只有對象才可以添加屬性。
alert(box.age); //不是引用類型,無法輸出;
</script>

簡而言之,堆內(nèi)存存放引用值,棧內(nèi)存存放固定類型值。

<script type="text/javascript">
  var man = new Object();//man指向了棧內(nèi)存的空間地址
  man.name = "Jack";
  var man2 = man;//man2獲得了man的指向地址

  alert(man2.name);//兩個都彈出Jack
  alert(man.name);
</script>

復(fù)制變量值

再看下面這個例子:

<script type="text/javascript">
  var man = new Object();//man指向了棧內(nèi)存的空間地址
  man.name = "Jack";
  var man2 = man;//man2獲得了man的指向地址

  man2.name = "ming";//因為他們都指向同一個object,同一個name,不管修改誰,大家都修改了
  alert(man2.name);//兩個都彈出ming
  alert(man.name);
</script>

由以上可以得出:在變量復(fù)制方面,基本類型和引用類型也有所不同,基本類型復(fù)制的是值本身,而引用類型復(fù)制的是地址。

傳遞參數(shù)

ECMAScript中,所有函數(shù)的參數(shù)都是按值傳遞的,

<script type="text/javascript">
   function box(num){   //按值傳遞
     num+=10;
     return num;
   }

   var num = 10;
   var result = box(num);
   alert(result); //如果是按引用傳遞,那么函數(shù)里的num會成為類似全局變量,把外面的number替換掉
   alert(num);  //也就是說,最后應(yīng)該輸出20(這里輸出10)
</script>

javascript沒有按引用傳遞的,如果存在引用傳遞的話,那么函數(shù)內(nèi)的變量將是全局變量,在外部也可以訪問。但這明顯是不可能的。

執(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)境是window對象,因此,所有的全局變量的函數(shù)都是作為window的屬性和方法創(chuàng)建的。

<script type="text/javascript">
   var name = "Jack";      //定義全局變量
   function setName(){
     return "trigkit4";
   }

   alert(window.name);    //全局變量,最外圍,屬于window屬性
   alert(window.setName()); //全局函數(shù),最外圍,屬于window方法
</script>

當(dāng)執(zhí)行環(huán)境內(nèi)的代碼執(zhí)行完畢后,該環(huán)境被銷毀,保存其中的變量和函數(shù)也隨之銷毀,如果是全局環(huán)境,需所有程序執(zhí)行完畢或網(wǎng)頁完畢后才會銷毀。

去掉var的局部變量

<script type="text/javascript">
   var name = "Jack";
   function setName(){
     name = "trigkit4";  //去掉var變成了全局變量
   }

   setName();
   alert(name);//彈出trigkit4
</script>

通過傳參,也是局部變量

<script type="text/javascript">
   var name = "Jack";
   function setName(name){  //通過傳參,也是局部變量
     alert(name);
   }

   setName("trigkit4");//彈出trigkit4
   alert(name);//彈出Jack
</script>

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

<script type="text/javascript">
   var name = "Jack";
   function setName(){
     function setYear(){  //setYear()方法的作用域在setName()內(nèi)
       return 21;
     }
   }
   alert(setYear());//無法訪問,出錯 
</script>

可以通過如下方法進行訪問:

<script type="text/javascript">
   var name = "Jack";
   function setName(){
     function setYear(){  //setYear()方法的作用域在setName()內(nèi)
       return 21;
     }
     return setYear();
   }
   alert(setName()); //彈出21
</script>

再一個作用域例子:

<script type="text/javascript">
   var name = "Jack";
   function setName(){
     function setYear(){  //setYear()方法的作用域在setName()內(nèi)
       var b = "hi";   //變量b的作用域在setYear()內(nèi)
       return 21;
     }
     alert(b);//無法訪問 
   }
</script>

當(dāng)代碼在一個環(huán)境中執(zhí)行的時候,就會形成一種叫做作用域鏈的東西,它的用途是保證對執(zhí)行環(huán)境中有訪問權(quán)限的變量和函數(shù)進行有序訪問(指按照規(guī)則層次來訪問),作用域鏈的前端,就是執(zhí)行環(huán)境的變量對象。

作用域

變量沒有在函數(shù)內(nèi)聲明或者聲明的時候沒有帶var就是全局變量,擁有全局作用域,window對象的所有屬性擁有全局作用域;在代碼任何地方都可以訪問,函數(shù)內(nèi)部聲明并且以var修飾的變量就是局部變量,只能在函數(shù)體內(nèi)使用,函數(shù)的參數(shù)雖然沒有使用var但仍然是局部變量。

沒有塊級作用域

沒有塊級作用域

// if語句:

<script type="text/javascript">
if(true){            //if語句的花括號沒有作用域的功能。

var box = "trigkit4";
}
alert(box);//彈出 trigkit4
</script>


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

變量的查詢

在變量的查詢中,訪問局部變量要比全局變量來得快,因此不需要向上搜索作用域鏈。
如下例子:

<script type="text/javascript">
   var name = "Jack";
   function setName(){
      var name = "trigkit4";
      return name; //從底層向上搜索變量
  }
  alert(setName());   
</script>

內(nèi)存問題

javascript具有自動垃圾回收機制,一旦數(shù)據(jù)不再使用,可以將其設(shè)為"null"來釋放引用

循環(huán)引用

  一個很簡單的例子:一個DOM對象被一個Javascript對象引用,與此同時又引用同一個或其它的Javascript對象,這個DOM對象可能會引發(fā)內(nèi)存泄露。這個DOM對象的引用將不會在腳本停止的時候被垃圾回收器回收。要想破壞循環(huán)引用,引用DOM元素的對象或DOM對象的引用需要被賦值為null。

閉包

在閉包中引入閉包外部的變量時,當(dāng)閉包結(jié)束時此對象無法被垃圾回收(GC)。

var a = function() {
 var largeStr = new Array(1000000).join('x');
 return function() {
  return largeStr;
 }
}();

DOM泄露

當(dāng)原有的COM被移除時,子結(jié)點引用沒有被移除則無法回收。

var select = document.querySelector;
var treeRef = select('#tree');

//在COM樹中l(wèi)eafRef是treeFre的一個子結(jié)點
var leafRef = select('#leaf'); 
var body = select('body');

body.removeChild(treeRef);

//#tree不能被回收入,因為treeRef還在
//解決方法:
treeRef = null;

//tree還不能被回收,因為葉子結(jié)果leafRef還在
leafRef = null;

//現(xiàn)在#tree可以被釋放了。

Timers計(定)時器泄露

定時器也是常見產(chǎn)生內(nèi)存泄露的地方:

for (var i = 0; i < 90000; i++) {
 var buggyObject = {
  callAgain: function() {
   var ref = this;
   var val = setTimeout(function() {
    ref.callAgain();
   }, 90000);
  }
 }

 buggyObject.callAgain();
 //雖然你想回收但是timer還在
 buggyObject = null;
}

調(diào)試內(nèi)存

Chrome自帶的內(nèi)存調(diào)試工具可以很方便地查看內(nèi)存使用情況和內(nèi)存泄露:
在 Timeline -> Memory 點擊record即可:

相關(guān)文章

  • JS同步、異步、延遲加載的方法

    JS同步、異步、延遲加載的方法

    本篇文章講述了JS同步、異步、延遲加載的方法,具有很好的參考價值,希望對大家的學(xué)習(xí)有所幫助
    2018-05-05
  • 深入分析JavaScript 事件循環(huán)(Event Loop)

    深入分析JavaScript 事件循環(huán)(Event Loop)

    這篇文章主要介紹了JavaScript 事件循環(huán)(Event Loop)的的相關(guān)資料,文中講解非常細致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • Javascript基礎(chǔ)教程之比較操作符

    Javascript基礎(chǔ)教程之比較操作符

    這篇文章主要介紹了Javascript基礎(chǔ)教程之比較操作符,需要的朋友可以參考下
    2015-01-01
  • js時間比較示例分享(日期比較)

    js時間比較示例分享(日期比較)

    這篇文章主要介紹了js時間比較示例,代碼簡單,運行后可以看到結(jié)果,需要的朋友可以參考下
    2014-03-03
  • JS嚴格模式知識點總結(jié)

    JS嚴格模式知識點總結(jié)

    本篇文章給大家整理了JS嚴格模式下的相關(guān)知識點以及代碼實例分享,感興趣的跟著小編一起學(xué)習(xí)下吧。
    2018-02-02
  • JavaScript門道之標準庫

    JavaScript門道之標準庫

    標準庫指的是js的標準內(nèi)置對象,是js這門語言本身初始時提供的在全局范圍的對象,需要的朋友可以參考下
    2018-05-05
  • JavaScript 操作宏任務(wù)與微任務(wù)

    JavaScript 操作宏任務(wù)與微任務(wù)

    這篇文章主要介紹了JavaScript 操作宏任務(wù)與微任務(wù),簡要說明了有關(guān)定時器和具體的代碼實現(xiàn),希望對你有所幫助
    2021-06-06
  • JWT構(gòu)成及工作原理

    JWT構(gòu)成及工作原理

    這篇文章介紹了JWT構(gòu)成及工作原理,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-01-01
  • JS中switch的四種寫法示例

    JS中switch的四種寫法示例

    這篇文章主要為大家介紹了JS中switch的四種寫法示例代碼,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-07-07
  • 圖解prototype、proto和constructor的三角關(guān)系

    圖解prototype、proto和constructor的三角關(guān)系

    在javascript中,prototype、constructor以及__proto__之間有著“著名”的剪不斷理還亂的三角關(guān)系,樓主就著自己對它們的淺顯認識,來粗略地理理以備忘,有不對之處還望斧正。
    2016-07-07

最新評論