閉包
指有權限訪問另一個函數作用域中的變量的函數,常見形式是函數內部嵌套函數。一般來講,當函數執行完畢后,局部活動對象就會被銷毀,內存中僅保存全局作用域,但是閉包的情況有所不同。因為外部函數的活動對象一直在被匿名函數中的作用域引用,即使執行環境的作用域被銷毀了,但是它的活動對象也會留在內存當中。直到匿名函數被銷毀后,外部函數的活動對象才會被銷毀
function f1 () {
var b = 1
return function f2 () {
alert(b)
}
}
var a = f1() //f1執行完畢,變量b未被回收,仍然被f2引用著。
a= null //清空對f2的引用,變量a指向null,a和函數f2之間的聯系被切斷。f2不再被引用,因此f2被回收,f2被回收后,它對b的引用也就不存在了,b不再被引用,b也被回收不再占用內存。
閉包會攜帶包含它的函數作用域,因此會比其他函數占用更多的內存。
面試題1:
var name = 'global'
var obj = {
name : 'obj',
dose : function(){
this.name = 'dose'
return function(){
return this.name;
}
}
}
alert(obj.dose().call(this)) //global
面試題2:
var name = 'global'
var obj = {
name : 'obj',
dose : function(){
this.name = 'dose'
return function(){
return this.name
}.bind(this)
}
}
alert(obj.dose().call(this)) //dose
面試題3:
var name = 'global'
var obj = {
name : 'obj',
dose : function(){
var that = this;
this.name = 'dose'
return function(){
return that.name
}
}
}
alert(obj.dose().call(this)) //dose
模仿塊級作用域
(function () {
//這里是塊級作用域
})()
如下:
function a() {
(function () {
for(var i = 0; i<count ;i++) {
alert(i)
}
})()
alert(i) //報錯
}//這種做法可以減少閉包占用內存的問題,因為沒有指向匿名函數的引用,只要函數執行完畢,就可以立即銷毀作用域鏈
私有變量
嚴格的說,js中沒有私有成員的概念,所有對象屬性都是公用的。但是有私有變量的概念,在函數中定義的變量都可以被認為是私有變量,因為不能在函數外部訪問這些變量。私有變量包括函數的參數,局部變量和函數內部定義的其他函數。
特權方法:指的是有權訪問私有變量和私有函數的共有方法。
function f1 () {
var a = 1
function f2 () {
return false
}
this.publicMethod = function (){//特權方法,想要訪問a和f2只能通過publicMethod訪問
a++
return f2()
}
}