天天看點

好程式員Java教程分享JavaScript常見面試題四

  好程式員Java教程分享JavaScript常見面試題四1、下面的代碼将輸出什麼到控制台,為什麼?

console.log(1 + "2" + "2");console.log(1 + +"2" + "2");console.log(1 + -"1" + "2");console.log(+"1" + "1" + "2");console.log( "A" - "B" + "2");console.log( "A" - "B" + 2);

上面的代碼将輸出以下内容到控制台:

"122""32""02""112""NaN2"NaN

原因是…

這裡的根本問題是,JavaScript(ECMAScript)是一種弱類型語言,它可對值進行自動類型轉換,以适應正在執行的操作。讓我們通過上面的例子來說明這是如何做到的。

例1:1 + "2" + "2" 輸出:"122" 說明: 1 + "2" 是執行的第一個操作。由于其中一個運算對象("2")是字元串,JavaScript會假設它需要執行字元串連接配接,是以,會将 1 的類型轉換為 "1", 1 + "2"結果就是 "12"。然後, "12" + "2" 就是 "122"。

例2: 1 + +"2" + "2" 輸出: "32" 說明:根據運算的順序,要執行的第一個運算是 +"2"(第一個 "2" 前面的額外 + 被視為一進制運算符)。是以,JavaScript将 "2" 的類型轉換為數字,然後應用一進制 + 号(即,将其視為一個正數)。其結果是,接下來的運算就是 1 + 2 ,這當然是 3。然後我們需要在一個數字和一個字元串之間進行運算(即, 3 和 "2"),同樣的,JavaScript會将數值類型轉換為字元串,并執行字元串的連接配接,産生 "32"。

例3: 1 + -"1" + "2" 輸出: "02" 說明:這裡的解釋和前一個例子相同,除了此處的一進制運算符是 - 而不是 +。先是 "1" 變為 1,然後當應用 - 時又變為了 -1 ,然後将其與 1相加,結果為 0,再将其轉換為字元串,連接配接最後的 "2" 運算對象,得到 "02"。

例4: +"1" + "1" + "2" 輸出: "112" 說明:雖然第一個運算對象 "1"因為字首的一進制 + 運算符類型轉換為數值,但又立即轉換回字元串,當連接配接到第二個運算對象 "1" 的時候,然後又和最後的運算對象"2" 連接配接,産生了字元串 "112"。

例5: "A" - "B" + "2" 輸出: "NaN2" 說明:由于運算符 - 不能被應用于字元串,并且 "A" 和 "B" 都不能轉換成數值,是以,"A" - "B"的結果是 NaN,然後再和字元串 "2" 連接配接,得到 "NaN2" 。

例6: "A" - "B" + 2 輸出: NaN 說明:參見前一個例子, "A" - "B" 結果為 NaN。但是,應用任何運算符到NaN與其他任何的數字運算對象,結果仍然是 NaN。

2、下面的遞歸代碼在數組清單偏大的情況下會導緻堆棧溢出。在保留遞歸模式的基礎上,你怎麼解決這個問題?

var list = readHugeList();var nextListItem = function() { var item = list.pop(); if (item) { // process the list item...

nextListItem();

}

};

潛在的堆棧溢出可以通過修改nextListItem 函數避免:

setTimeout( nextListItem, 0);

堆棧溢出之是以會被消除,是因為事件循環操縱了遞歸,而不是調用堆棧。當 nextListItem 運作時,如果 item不為空,timeout函數(nextListItem)就會被推到事件隊列,該函數退出,是以就清空調用堆棧。當事件隊列運作其timeout事件,且進行到下一個 item 時,定時器被設定為再次調用 extListItem。是以,該方法從頭到尾都沒有直接的遞歸調用,是以無論疊代次數的多少,調用堆棧保持清空的狀态。

3、JavaScript中的“閉包”是什麼?請舉一個例子。

閉包是一個可以通路外部(封閉)函數作用域鍊中的變量的内部函數。閉包可以通路三種範圍中的變量:這三個範圍具體為:(1)自己範圍内的變量,(2)封閉函數範圍内的變量,以及(3)全局變量。

下面是一個簡單的例子:

var globalVar = "xyz";

(function outerFunc(outerArg) { var outerVar = 'a';

(function innerFunc(innerArg) { var innerVar = 'b'; console.log( "outerArg = " + outerArg + "n" + "innerArg = " + innerArg + "n" + "outerVar = " + outerVar + "n" + "innerVar = " + innerVar + "n" + "globalVar = " + globalVar);

})(456);

})(123);

在上面的例子中,來自于 innerFunc, outerFunc和全局命名空間的變量都在 innerFunc的範圍内。是以,上面的代碼将輸出如下:

outerArg = 123innerArg = 456outerVar = ainnerVar = bglobalVar = xyz

4、下面的代碼将輸出什麼:

for (var i = 0; i < 5; i++) {

setTimeout(function() { console.log(i); }, i * 1000 );

解釋你的答案。閉包在這裡能起什麼作用?

上面的代碼不會按預期顯示值0,1,2,3,和4,而是會顯示5,5,5,5,和5。

原因是,在循環中執行的每個函數将整個循環完成之後被執行,是以,将會引用存儲在 i中的最後一個值,那就是5。

閉包可以通過為每次疊代建立一個唯一的範圍,存儲範圍内變量的每個唯一的值,來防止這個問題,如下:

(function(x) {

setTimeout(function() { console.log(x); }, x * 1000 );

})(i);

這就會按預期輸出0,1,2,3,和4到控制台。