「前端 · 面試」記錄一些有意思的題目(二)

本文記錄一些在網(wǎng)上看到的昔逗,我認為有意思的前端面試題枉长,歡迎轉載,轉載請注明出處:林東洲的博客 | Lindz Blog钢属。

輸出下面程序題的結果:

function Foo() {
    getName = function () { alert (1); };
    return this;
}
Foo.getName = function () { alert (2);};
Foo.prototype.getName = function () { alert (3);};
var getName = function () { alert (4);};
function getName() { alert (5);}

//請寫出以下輸出結果:
Foo.getName();
getName();
Foo().getName();
getName();
new Foo.getName();
new Foo().getName();
new new Foo().getName();

先來理解下題目,首先創(chuàng)建了一個Foo()函數(shù)门躯,然后在給Foo函數(shù)增添一個函數(shù)屬性getName,在Foo原型上增添一個getName函數(shù)屬性淆党,然后再以函數(shù)表達式和函數(shù)聲明分別創(chuàng)建了一個同名函數(shù)getName。

第一問:Foo.getName()返回2讶凉。

第二問:首先需要知道函數(shù)表達式和函數(shù)聲明的區(qū)別染乌。

函數(shù)聲明具有變量聲明提升(即所有聲明變量或聲明函數(shù)都會被提升到當前函數(shù)的頂部)

例:

console.log('x' in window);
console.log(typeof func);
var x = 0;
function func(){}

js引擎會將其解析為:

var x;
function func(){}
console.log('x' in window);    //true
console.log(typeof func);      //function
x = 0;

如果能明白的話就能理解下面代碼的輸出:

console.log(x);     //function(){}
var x = 1;          
console.log(x);     //1
function x(){}
console.log(x);     //1

所以js引擎在解析代碼時候就變成了:

//省略部分代碼
function Foo() {
    getName = function () { alert (1); };
    return this;
}
var getName;
function getName() { alert (5);}
getName = function () { alert (4);};

getName();       //4

后面函數(shù)表達式的賦值將前面的getName值個覆蓋了,所以第二問輸出4懂讯。

第三問:首先執(zhí)行Foo()函數(shù)荷憋,并執(zhí)行getName的賦值語句。

它會查找當前作用域上是否存在這個變量褐望,如果沒有就向它上層作用域繼續(xù)尋找勒庄,直到找到window全局作用域,若還是沒有則在全局作用域上創(chuàng)建getName變量瘫里,

因為window域中存在著getName這個變量即(var getName = function () { alert (4);};)实蔽,所以將其替換為(getName = function () { alert (1); };)
并返回window對象,所以Foo().getName()的結果為1谨读;

若代碼改為:

function Foo() {
    var getName = function () { alert (1); };
    return this;
}
Foo().getName();     //TypeError: Foo().getName is not a function. 

創(chuàng)建getName時候多加一個var局装,即在當前作用域中添加一個局部變量getName,而在執(zhí)行完畢后該局部變量就會被回收劳殖,并且該函數(shù)返回window全局對象贼邓,故在訪問window中的getName會報錯,因為沒有創(chuàng)建window對象無此變量闷尿。

第四問:正如上一問塑径,因為全局作用于window中getName被替換。

即(getName = function () { alert (1); };)
故此時getName();輸出結果為1填具。

第五問 new Foo.getName(); ,此處考察的是js的運算符優(yōu)先級問題统舀。

根據(jù)圖中優(yōu)先級的描述:

優(yōu)先級圖

可以得知:可以得知點(.)的優(yōu)先級高于new操作,所以原式相當于是:

new (Foo.getName)();

所以結果為2劳景;

第六問 new Foo().getName()

根據(jù)表的執(zhí)行結果為:(new Foo()).getName();

所以接下來先討論一下js構造函數(shù)返回值的問題:

js返回值分為三類:
1誉简、沒有返回值則創(chuàng)建一個空對象,并把函數(shù)中與this綁定的屬性添加盟广;

function F(){ this.name = 'foo';}
var obj = new F();   // obj: {name:'foo'} 創(chuàng)建一個對象闷串,并綁定函數(shù)中this所綁定的屬性

2、若有返回值則檢查返回值的類型筋量,若返回值為非引用類型烹吵,比如基本類型:(string,number,boolean,null,undefined)碉熄,則js引擎會忽略該返回值,并且創(chuàng)建一個新對象肋拔,并把函數(shù)中與this綁定的屬性添加锈津;

function F(){ this.name = 'foo'; return true;}
var obj = new F();   // obj: {name:'foo'} 創(chuàng)建一個對象,并綁定函數(shù)中this所綁定的屬性

3凉蜂、若返回值是引用類型琼梆,則實際返回值為這個引用類型。

function F(){this.name = 'foo'; return {x:1};}
var obj = new F();    //obj: {x:1}

如果理解這些窿吩,再來看下面一個例子:

var obj;
function F1(){this.name = 'foo'; return 'a';}
obj = new F1();    //obj: F1{name: 'foo'}

function F2(){this.name = 'foo'; return String('a');}
obj = new F2();    //obj: F2{name: 'foo'}

function F3(){this.name = 'foo'; return new String('a');}
obj = new F3();    //obj: String {0: "a", length: 1}

從例子中我們可以得出:無論 ‘a(chǎn)’還是String('a')創(chuàng)建的都是基本類型值茎杂,而new String('a')則創(chuàng)建一個字符串包裝對象。

原題中纫雁,返回的是this蛉顽,而this在構造函數(shù)中本來就代表當前實例化對象。
之后調(diào)用實例化對象的getName函數(shù)先较,因為在Foo構造函數(shù)中沒有為實例化對象添加任何屬性携冤,遂到當前對象的原型對象(prototype)中尋找getName,故最終輸出3闲勺。

第七問:new new Foo().getName(); 同樣是運算符優(yōu)先級問題曾棕。

執(zhí)行結果為:new ((new Foo()).getName)();
先初始化Foo的實例化對象,然后將其原型上的getName函數(shù)作為構造函數(shù)再次new菜循,故最終結果為3

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末翘地,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子癌幕,更是在濱河造成了極大的恐慌衙耕,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,599評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡鹦倚,警方通過查閱死者的電腦和手機院刁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,629評論 3 385
  • 文/潘曉璐 我一進店門板驳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事『汪ぃ” “怎么了?”我有些...
    開封第一講書人閱讀 158,084評論 0 348
  • 文/不壞的土叔 我叫張陵碟刺,是天一觀的道長锁保。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么爽柒? 我笑而不...
    開封第一講書人閱讀 56,708評論 1 284
  • 正文 為了忘掉前任吴菠,我火速辦了婚禮,結果婚禮上霉赡,老公的妹妹穿的比我還像新娘。我一直安慰自己幔托,他們只是感情好穴亏,可當我...
    茶點故事閱讀 65,813評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著重挑,像睡著了一般嗓化。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上谬哀,一...
    開封第一講書人閱讀 50,021評論 1 291
  • 那天刺覆,我揣著相機與錄音,去河邊找鬼史煎。 笑死谦屑,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的篇梭。 我是一名探鬼主播氢橙,決...
    沈念sama閱讀 39,120評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼恬偷!你這毒婦竟也來了悍手?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,866評論 0 268
  • 序言:老撾萬榮一對情侶失蹤袍患,失蹤者是張志新(化名)和其女友劉穎坦康,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體诡延,經(jīng)...
    沈念sama閱讀 44,308評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡滞欠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,633評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了肆良。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片仑撞。...
    茶點故事閱讀 38,768評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖妖滔,靈堂內(nèi)的尸體忽然破棺而出隧哮,到底是詐尸還是另有隱情,我是刑警寧澤座舍,帶...
    沈念sama閱讀 34,461評論 4 333
  • 正文 年R本政府宣布沮翔,位于F島的核電站,受9級特大地震影響,放射性物質發(fā)生泄漏采蚀。R本人自食惡果不足惜疲牵,卻給世界環(huán)境...
    茶點故事閱讀 40,094評論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望榆鼠。 院中可真熱鬧纲爸,春花似錦、人聲如沸妆够。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,850評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽神妹。三九已至颓哮,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間鸵荠,已是汗流浹背冕茅。 一陣腳步聲響...
    開封第一講書人閱讀 32,082評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蛹找,地道東北人姨伤。 一個月前我還...
    沈念sama閱讀 46,571評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像庸疾,于是被迫代替她去往敵國和親姜挺。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,666評論 2 350

推薦閱讀更多精彩內(nèi)容