關(guān)于回調(diào)函數(shù)對(duì)于初學(xué)者來說是個(gè)比較難以理解的知識(shí)點(diǎn)。本文以內(nèi)存(只以堆棧)的角度來理解回調(diào)函數(shù)
變量的內(nèi)存情況
在js中數(shù)據(jù)類型的劃分可以分為基本類型與引用類型。兩種類型在內(nèi)存中的表現(xiàn)形式是不一樣的
1、基本類型
```js
//棧區(qū)創(chuàng)建一個(gè)變量 記錄的值為10
var num = 10;
//棧區(qū)創(chuàng)建一個(gè)n變量 并且賦值為num變量的值
var n = num;
//將棧區(qū)n變量的值修改為20
n = 20;
console.log(num)
```
2、引用類型
```js
//棧區(qū)創(chuàng)建一個(gè)arr變量 將具體的元素值保存到堆區(qū) 并且在arr變量中的值記錄為堆區(qū)的地址
var arr = [1,2];
//棧區(qū)創(chuàng)建一個(gè)newArray的變量 并且將 arr在棧區(qū)存儲(chǔ)的地址賦值給newArray
var newArray = arr;
//順著newArray記錄的地址值修改數(shù)據(jù)
newArray[0] = 100;
console.log(arr);// [100,2]
```
函數(shù)的情況
函數(shù)也是一種引用類型,所以創(chuàng)建一個(gè)函數(shù)就與創(chuàng)建數(shù)組在內(nèi)存的情況類似,當(dāng)創(chuàng)建一個(gè)函數(shù)時(shí),會(huì)在棧區(qū)創(chuàng)建一個(gè)變量,然后將函數(shù)代碼相關(guān)的信息存儲(chǔ)到堆區(qū),最后將堆區(qū)的地址賦值給棧區(qū)的變量保存
```js
function fn(){
console.log('fn')
}
// 創(chuàng)建函數(shù) 就是創(chuàng)建了一個(gè)函數(shù)名稱的同名變量。并且值指向"堆區(qū)”的地址(聲明式與賦值式一致).在堆區(qū)中會(huì)保存代碼片段相關(guān)的信息
//上面的代碼等價(jià)于 var fn = function(){consoel.log('fn')}
console.log(fn)
// 當(dāng)函數(shù)調(diào)用時(shí),系統(tǒng)可以通過"變量名稱/函數(shù)名稱" 找到函數(shù)代碼在堆區(qū)的地址。也就是可以改寫形式為“內(nèi)存地址()”,()就是表示要將代碼運(yùn)行起來
fn()
```
自調(diào)用匿名函數(shù)
```js
(function(){
console.log('123456')
})();
/*
function(){
console.log('123456')
}
就是一個(gè)函數(shù)表達(dá)式,本身就可以得到一個(gè)內(nèi)存地址 例如該內(nèi)存地址等于0X111 ,內(nèi)存地址中存儲(chǔ)著函數(shù)的代碼信息
所以上面的代碼可以理解為(0X111)() 這種形式,這種形式與 賦值式的函數(shù)時(shí)是一直 所以最后可以調(diào)用函數(shù)
*/
```
回調(diào)函數(shù)
回調(diào)函數(shù)就是將函數(shù)作為作為另外一個(gè)函數(shù)調(diào)用的參數(shù)進(jìn)行傳遞。其本質(zhì)也只是傳遞了一個(gè)內(nèi)存地址
1、函數(shù)參數(shù)傳遞引用類型
```js
function fn(a){
//3、當(dāng)函數(shù)調(diào)用時(shí) 會(huì)隱藏的進(jìn)行 var a = arr的操作 所以將arr變量存儲(chǔ)的地址賦值給了a變量 所以a就是一個(gè)地址 因?yàn)閍變量存儲(chǔ)的值對(duì)應(yīng)的是函數(shù)所以最終時(shí)候可以通過a[下標(biāo)]訪問元素的
console.log(a)
}
//1、棧區(qū)創(chuàng)建一個(gè)arr變量 賦值為堆區(qū)中保存數(shù)據(jù)的地址
var arr = [1,2,3]
//2、調(diào)用函數(shù)并且傳遞arr變量的值 傳遞的是一個(gè)內(nèi)存地址
fn(arr)
```
2、將函數(shù)作為參數(shù)傳遞
```js
//1、創(chuàng)建一個(gè)fn變量 并且賦值為堆區(qū)中的地址
function fn(a){
//4、執(zhí)行函數(shù)時(shí)進(jìn)行形參賦值 var a = callback 所以將callback變量的地址賦值給了a 由于a是一個(gè)函數(shù) 所以可以()
a()
}
//2、創(chuàng)建callback 并且賦值為堆區(qū)中的地址
var callback = function(){
console.log('回調(diào)函數(shù)');
}
//3、調(diào)用fn函數(shù) 并且傳遞參數(shù)callback的地址
fn(callback)
```
更多關(guān)于“web前端培訓(xùn)”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓(xùn)經(jīng)驗(yàn),課程大綱更科學(xué)更專業(yè),有針對(duì)零基礎(chǔ)的就業(yè)班,有針對(duì)想提升技術(shù)的提升班,高品質(zhì)課程助理你實(shí)現(xiàn)夢(mèng)想。