摘要:題目二答案會報錯未定義這段代碼中混合了函數聲明和函數表達式的形式,而函數實際上是綁定到了上而不是。除此之外函數聲明與函數表達式的語法其實是等價的。因此,在外層函數函數體內的兩個函數聲明,都會提升到之前執行。
這是我在Javascript微信公眾號上看到的一篇文章,覺得挺有意思的,所以轉載過來跟大家分享一下,同時,對這些題目也加上了一些我個人的理解,如果有不對的地方,請大家指正。
題目一
(function(){ return typeof arguments; })();
答案:Object
typeof所返回的可能值為:number、string、boolean、undefined、object、function。而我們知道arguments是一個偽數組,而偽數組也是一個對象,所以答案應該是Object。那么,如果把typeof arguments改成arguments instanceof Array呢?
可以看到,返回的是false。因此,偽數組不是一個數組,只有在使用了Array.prototype.slice.call()進行了轉換后,才能轉化成一個真數組。
題目二
var f = function g(){ return 23; }; typeof g();
答案:會報錯(g()未定義)
這段代碼中混合了函數聲明和函數表達式的形式,而函數實際上是綁定到了f上而不是g。但是這么寫是否符合Javascript的要求呢?在《Javascript高級程序設計》中:
也可以同時使用函數聲明和函數表達式,例如:var sum = function sum(){}。不過,這種語法在safari中會導致錯誤。
既然在除了safari以外的瀏覽器可以同時使用函數聲明和函數表達式來聲明函數,那么為什么題目中的代碼會報錯呢?我的理解是,這樣寫的結果是導致了函數名g被包裝進了f中,變成了f的一個局部變量,只能在f函數內部訪問到g,我把代碼修改了一下:
var f = function g(){ alert(g); return 23; }; f();
我在f函數內部訪問g,彈框彈出的內容為function g(){ alert(g);return 23; },沒有報錯并且訪問成功,g的函數體內容與f是一樣的,只是函數名不一樣。
————————我是一條分割線——————————
我們再來回顧一下函數聲明與函數表達式吧(因為在之前的一次面試中被問到了它們的區別,我一下沒打上來……所以還是增強一下記憶)
函數聲明的形式:
function funcName(){ /*some code*/}
函數表達式的形式:
var funcName = function(){/*some code*/}
在《Javascript高級程序設計》中有一段話:
解析器在向執行環境中加載數據時,對函數聲明和函數表達式并非一視同仁的。解析器會率先讀取函數聲明,并使其在執行任何代碼之前可用(可以訪問);至于函數表達式,則必須等到解析器執行到它所在的代碼行,才會真正被解釋執行。
也就是說,在函數聲明的代碼段之前訪問函數,是能夠訪問到的(函數聲明提升)。而在函數表達式的代碼段之前訪問函數,則會發生錯誤,因為在執行到函數所在語句之前,函數表達式聲明的變量名中不會保存對函數的的引用。
除此之外函數聲明與函數表達式的語法其實是等價的。
題目三:
(function(x){ delete x; return x; })(1);
答案:1
注意:delete是用來刪除對象的屬性,無法刪除普通變量。
題目四
var y = 1, x = y = typeof x; x;
答案:undefined
因為賦值運算是從右往左的,因此題目中的代碼展開來等價于:
var y = 1; y = typeof x; var x = y; x;
第二行代碼在訪問x時,x并沒有聲明,值為undefined,所以typeof x也就是undefined了。
題目五
(function f(f){ return typeof f(); })(function(){ return 1; });
答案:number
把代碼分開來看會比較清楚:
var baz = function(){ return 1; }; (function f(f){ return typeof f(); })(baz);
先將function(){ return 1; }賦值給baz,則baz()應該是1。再把baz當成參數傳入到立即執行函數中,立即執行函數返回的是傳入的參數的類型,即baz()的類型,baz()返回1類型為number。
題目六
var foo = { bar: function() { return this.baz; }, baz: 1 }; (function(){ return typeof arguments[0](); })(foo.bar);
答案:undefined
這里關鍵要理解this的所指。arguments[0]即foo.bar,而函數中的this指的就是調用arguments[0]()這個函數的對象,即window對象。但是window對象并沒有baz這個屬性,因此,返回的是undefined。
題目七
var foo = { bar: function(){ return this.baz; }, baz: 1 } typeof (f = foo.bar)();
答案:undefined
將代碼改寫一些變成:
var foo = { bar: function(){ return this.baz; }, baz: 1 } var f = foo.bar; typeof f();
foo.bar是一個函數,將foo.bar賦值給了f,那么f變成了一個函數名。直接調用f()時,調用它的對象為window,而window中并沒有baz屬性,所以輸出的是undefined。
題目八
var f = (function f(){ return "1"; }, function g(){ return 2; })(); typeof f;
答案:number
在這里,我們需要了解一下逗號操作符的作用:
在用于賦值時,逗號操作符總會返回表達式中的最后一項。
所以,這個立即執行函數返回的應該是后面那個函數的返回值,返回值為2,類型為number。
題目九
var x = 1; if (function f(){}) { x += typeof f; } x;
答案:1undefined
首先,第一個問題是:函數在判斷語句中判斷的結果是true還是false?,我們可以運行一下以下代碼:
Boolean(function f(){});//true
將一個空函數轉換為布爾型,結果為true,所以題中的代碼是能夠進入if語句的。那么第二個問題是:為什么typeof f會是undefined?因為if判斷條件中并不能真正的聲明一個函數,因此,f是一個沒有被聲明過的函數名,它的類型為undefined。
題目十
(function f(){ function f(){ return 1; } return f(); function f(){ return 2; } })();
答案:2
我們在題目四里說過:Javascript解析器會率先讀取函數聲明,使它在任何代碼之間可用。因此,在外層函數f函數體內的兩個函數聲明,都會提升到return f()之前執行。而又因為Javascript沒有重載,因此以最后執行的同名函數輸出結果為準。
題目十一
function f(){ return f; } new f() instanceof f;
答案:false
1、首先我們要了解instanceof的作用:instanceof是用來檢測變量是否是給定引用類型的實例。
2、然后再看f()函數體中return f,返回了自己。
3、通過new來調用構造函數生成實例的過程為:1)創建一個空對象;2)將類的prototype中的屬性和方法復制到實例中;3)將第一步創建的空對象做為類的參數調用類的構造函數;4)返回一個新的實例對象。
然而,在以new形式調用構造函數f()時,f()返回了自身,將new返回的實例對象覆蓋掉了,因此,new之后的結果不是f的一個引用了。
題目十二
var x = [typeof x, typeof y][1]; typeof typeof x;
答案:string
很明顯的可以看出typeof x = undefined和typeof y = undefined,那么x是undefined。其實其他的都不看,光看typeof typeof就能確定輸出的是string了。
題目十三
(function(foo){ return typeof foo.bar; })({ foo: { bar: 1 } });
答案:undefined
這道題目有一個小陷阱,一不注意還真被忽悠進去了。我們仔細看一個參數傳入的內容最外面有一對{},那么我們把題目修改一下就一目了然了:
var temp = { foo: { bar: 1 } }; (function(foo){ return typeof foo.bar; })(temp);
傳入的參數為temp這個對象,因此函數中的foo.bar實際上為temp.bar,但是temp里只有foo一個屬性,沒有bar,所以應該是undefined。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/91727.html
摘要:虛擬代理延遲執行虛擬代理的目的,是將開銷大的運算延遲到需要時再執行。 showImg(https://segmentfault.com/img/bVbuitm?w=800&h=600); 代理模式:為一個對象提供一個代用品或占位符,以便控制它的訪問。 當我們不方便直接訪問某個對象時,或不滿足需求時,可考慮使用一個替身對象來控制該對象的訪問。替身對象可對請求預先進行處理,再決定是否轉交給...
摘要:先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學,大學期間開始自學前端開發,在今年春招實習和秋招的時候投了一些公司,拿到一些京東拼多多虎牙等,總體來說還算滿意,特地寫一篇文章來總結一下面試的那些套路。 showImg(https://segmentfault.com/img/remote/1460000011897700); 先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學...
摘要:先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學,大學期間開始自學前端開發,在今年春招實習和秋招的時候投了一些公司,拿到一些京東拼多多虎牙等,總體來說還算滿意,特地寫一篇文章來總結一下面試的那些套路。 showImg(https://segmentfault.com/img/remote/1460000011897700); 先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學...
摘要:先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學,大學期間開始自學前端開發,在今年春招實習和秋招的時候投了一些公司,拿到一些京東拼多多虎牙等,總體來說還算滿意,特地寫一篇文章來總結一下面試的那些套路。 showImg(https://segmentfault.com/img/remote/1460000011897700); 先介紹一下本人應屆前端開發一枚,非科班出身,專業是化學...
摘要:先實現棧操作遍歷鏈表,把每個節點都進中然后再遍歷鏈表,同時節點依次出棧,二者進行比較。 ?作者簡介:大家好,我是車神哥,府學路18號的車神? ?個人主頁:應無...
閱讀 2072·2023-04-25 22:58
閱讀 1419·2021-09-22 15:20
閱讀 2705·2019-08-30 15:56
閱讀 1996·2019-08-30 15:54
閱讀 2113·2019-08-29 12:31
閱讀 2736·2019-08-26 13:37
閱讀 601·2019-08-26 13:25
閱讀 2103·2019-08-26 11:58