国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

Javascript閉包用法實例分析

2019-11-20 13:20:06
字體:
來源:轉載
供稿:網友

本文實例分析了Javascript閉包的概念及用法。分享給大家供大家參考。具體如下:

提到閉包,想必大家都早有耳聞,下面說下我的簡單理解。
說實話平時工作中實際手動寫閉包的場景并不多,但是項目中用到的第三方框架和組件或多或少用到了閉包。
所以,了解閉包是非常必要的。呵呵...

一、什么是閉包

簡而言之,就是能夠讀取其他函數內部變量的函數。
由于JS變量作用域的特性,外部不能訪問內部變量,內部可以外部變量。

二、使用場景

1. 實現私有成員。
2. 保護命名空間,避免污染全局變量。
3. 緩存變量。

先看一個封裝的例子:

復制代碼 代碼如下:
var person = function () {
    // 變量作用域為函數內部,外部無法訪問
    var name = "default";

    return {
        getName: function () {
            return name;
        },
        setName: function (newName) {
            name = newName;
        }
    }
}();

console.log(person.name); // 直接訪問,結果為:undefined
console.log(person.getName()); // 結果為:default
console.log(person.setName("langjt"));
console.log(person.getName()); // 結果為:langjt

再看循環中常用閉包解決引用外部變量問題:

復制代碼 代碼如下:
var aLi = document.getElementsByTagName('li');
for (var i=0, len=aLi.length; i<len; i++) {
   aLi[i].onclick = function() {
     alert(i); // 無論點擊哪個<li>元素,彈出的值都為len,表明這里的i和在for之后打印i的值是一樣的。
   };
}

使用閉包后:
復制代碼 代碼如下:
var aLi = document.getElementsByTagName('li');
for (var i=0, len=aLi.length; i<len; i++) {
  aLi[i].onclick = (function(i) {
    return function() {
      alert(i); // 此時點擊<li>元素,就會彈出對應的下標了。
    }
  })(i);
}

三、注意事項

1. 內存泄漏

由于閉包會使得函數中的變量都被保存在內存中,內存消耗很大,所以不能濫用閉包,否則會造成網頁的性能問題。
比如:

復制代碼 代碼如下:
function foo() {
   var oDiv = document.getElementById(‘J_DIV');
   var id = oDiv.id;
   oDiv.onclick = function() {
     // alert(oDiv.id); 這里存在循環引用,IE低版本頁面關閉后oDiv仍在內存中。所以盡可能緩存基本類型而不是對象。
     alert(id);
   };
   oDiv = null;
}

2. 變量命名

如果內部函數的變量和外部函數的變量名相同時,那么內部函數再也無法指向外部函數那個同名的變量。
比如:

復制代碼 代碼如下:
function foo(num) {
  return function(num) {
    console.log(num);
  }
}
var f = new foo(9);
f(); // undefined

其實上面的用法,專業術語叫函數柯里化(Currying),就是把接受多個參數的函數變換成接受一個單一參數(最初函數的第一個參數)的函數,并且返回接受余下的參數而且返回結果的新函數的技術。本質上也利用了閉包可以緩存的特性,比如:

復制代碼 代碼如下:
var adder = function(num) {
    return function(y) {
        return num+y;
    };
};

var inc = adder(1);
var dec = adder(-1);
//inc, dec現在是兩個新的函數,作用是將傳入的參數值 (+/

主站蜘蛛池模板: 遂溪县| 新龙县| 抚远县| 额济纳旗| 衡东县| 泸州市| 芜湖市| 东兴市| 宜兴市| 常宁市| 名山县| 迭部县| 招远市| 湖南省| 米脂县| 晋州市| 大理市| 衡东县| 和田县| 奉新县| 高青县| 乌兰县| 公安县| 云南省| 涞水县| 仙游县| 苍梧县| 广安市| 都江堰市| 兴安县| 淮安市| 通州市| 东丽区| 吴旗县| 吴川市| 柳林县| 洞口县| 海南省| 平昌县| 屏山县| 德庆县|