JavaScript填坑史

說明: 這是筆者平時(shí)積累的一些覺得比較有意思或是比較有難度的JavaScript題目理解和心得巡验,會(huì)保持長(zhǎng)期更新。

個(gè)人博客:Damonare的個(gè)人博客

1.setTimeout和setInterval深入理解

setTimeout和setInterval深入理解這篇博客里筆者曾做過總結(jié),我們知道JavaScript試單線程的產(chǎn)物企巢,兩個(gè)函數(shù)就是利用了插入代碼的方式實(shí)現(xiàn)了偽異步漩氨,和AJAX的原理實(shí)際上是一樣的。下面來看下這個(gè)例子:

console.log("1");
setTimeout(function(){
        console.log("3")
    },0);
    console.log("2");

結(jié)果:控制臺(tái)依次輸出1,2,3;

 
function fn() { 
setTimeout(function(){alert('can you see me?');},1000); 
while(true) {} 
} 

你覺得這段代碼的執(zhí)行結(jié)果是什么呢串述?答案是,alert永遠(yuǎn)不會(huì)出現(xiàn)寞肖。
這是為什么呢纲酗?因?yàn)椋瑆hile這段代碼沒有執(zhí)行完新蟆,所以插入在后面的代碼便永遠(yuǎn)不會(huì)執(zhí)行觅赊。
綜上所述,其實(shí)JS終歸是單線程產(chǎn)物琼稻。無論如何“異步”都不可能突破單線程這個(gè)障礙吮螺。所以許多的“異步調(diào)用”(包括Ajax)事實(shí)上也只是“偽異步”而已。只要理解了這么一個(gè)概念,也許理解setTimeout和setInterval也就不難了鸠补。

2. 閉包初探小題

JavaScript閉包初探這篇博客里面進(jìn)行了初步探討萝风,有幾個(gè)小題個(gè)人覺得還是比較有意思的:

  var name = "The Window";
  var object = {
    name : "My Object",
    getNameFunc : function(){
      return function(){
        return this.name;
      };
    }
  };
  alert(object.getNameFunc()());//The Window
   var name = "The Window";
  var object = {
    name : "My Object",
    getNameFunc : function(){
      var that = this;
      return function(){
        return that.name;
      };
    }
  };
  alert(object.getNameFunc()());//My Object
function fun(n,o) {
  console.log(o)
  return {
    fun:function(m){
      return fun(m,n);
    }
  };
}
var a = fun(0);  a.fun(1);  a.fun(2);  a.fun(3);//undefined,?,?,?
var b = fun(0).fun(1).fun(2).fun(3);//undefined,?,?,?
var c = fun(0).fun(1);  c.fun(2);  c.fun(3);//undefined,?,?,?

//問:三行a,b,c的輸出分別是什么?

這是一道非常典型的JS閉包問題紫岩。其中嵌套了三層fun函數(shù)规惰,搞清楚每層fun的函數(shù)是那個(gè)fun函數(shù)尤為重要。

//答案:
//a: undefined,0,0,0
//b: undefined,0,1,2
//c: undefined,0,1,1

3. Array/map,Number/parseInt

["1", "2", "3"].map(parseInt)//求輸出結(jié)果

首先, map接受兩個(gè)參數(shù), 一個(gè)回調(diào)函數(shù) callback, 一個(gè)回調(diào)函數(shù)的this值
其中回調(diào)函數(shù)接受三個(gè)參數(shù) currentValue, index, arrary;而題目中, map只傳入了回調(diào)函數(shù)--parseInt.其次, parseInt 只接受兩個(gè)兩個(gè)參數(shù) string, radix(基數(shù)). radix的合法區(qū)間是2-36. 0或是默認(rèn)是10.所以本題即問

parseInt('1', 0);
parseInt('2', 1);
parseInt('3', 2);

后兩者參數(shù)不合法.所以答案是:[1, NaN, NaN]泉蝌;

4. 0.1+0.2!=0.3和9999999999999999 == 10000000000000000;

根據(jù)語言規(guī)范歇万,JavaScript 采用“IEEE 754 標(biāo)準(zhǔn)定義的雙精度64位格式”("double-precision 64-bit format IEEE 754 values")表示數(shù)字。據(jù)此我們能得到一個(gè)有趣的結(jié)論勋陪,和其他編程語言(如 C 和 Java)不同贪磺,JavaScript 不區(qū)分整數(shù)值和浮點(diǎn)數(shù)值,所有數(shù)字在 JavaScript 中均用浮點(diǎn)數(shù)值表示诅愚,所以在進(jìn)行數(shù)字運(yùn)算的時(shí)候要特別注意寒锚。精度丟失看看下面的例子:

0.1 + 0.2 = 0.30000000000000004

在具體實(shí)現(xiàn)時(shí),整數(shù)值通常被視為32位整型變量呻粹,在個(gè)別實(shí)現(xiàn)(如某些瀏覽器)中也以32位整型變量的形式進(jìn)行存儲(chǔ)壕曼,直到它被用于執(zhí)行某些32位整型不支持的操作,這是為了便于進(jìn)行位操作等浊。大整數(shù)精度在2的53次方以內(nèi)是不會(huì)丟失的腮郊,也就是說瀏覽器能精確計(jì)算Math.pow(2,53)以內(nèi)所有的數(shù),小數(shù)精度筹燕,當(dāng)十進(jìn)制小數(shù)的二進(jìn)制表示的有限數(shù)字不超過 52 位時(shí)轧飞,在 JavaScript 里也是可以精確存儲(chǔ)的。
解決辦法:Math.round( (.1+.2)*100)/100;

5. [1>2>3,3>2>1]

此題會(huì)讓人誤以為是2>1&&2<3,其實(shí)不是的撒踪,這個(gè)題等價(jià)于

1<2=>true;
true<3=>1<3=>true;
3<2=>true;
false<1=>0<1=>true;

答案:[true,true]
這個(gè)題的重點(diǎn)是對(duì)于運(yùn)算符的理解过咬,一是javascript對(duì)于不同類型數(shù)值的比較規(guī)則,詳見js比較表,javascript相等性判斷制妄;二是對(duì)于比較操作符賦值運(yùn)算符的理解掸绞,即一個(gè)自左向右一個(gè)自右向左~

6. 瀏覽器懵逼史(1)

3.tostring;
3..toString;
3...toString;

這個(gè)題感覺腦洞很大啊~先說答案:error,'3',error;
可如果是

var a=3;
a.toString;

卻又合法了答案就是'3';
為啥呢?
因?yàn)樵贘S中1.1,1.,.1都是合法數(shù)字案獭衔掸!那么在解析3.toString的時(shí)候到底是這是個(gè)數(shù)字呢,還是方法調(diào)用呢俺抽?瀏覽器就懵逼了唄敞映,只能拋出一個(gè)error,所以說感覺此題就是在戲耍瀏覽器......

7. 聲明提升

var name = 'World!';
(function () {
    if (typeof name === 'undefined') {
        var name = 'Jack';
        console.log('Goodbye ' + name);
    } else {
        console.log('Hello ' + name);
    }
})();

答案是什么呢...筆者第一次做的時(shí)候傻傻的覺得是Hello,world...實(shí)則不然,正確答案是:Goodbye Jack;
為什么呢磷斧,聲明提升...上述代碼相當(dāng)于下面的代碼:

var name = 'World!';
(function () {
    var name;
    if (typeof name === 'undefined') {
        name = 'Jack';
        console.log('Goodbye ' + name);
    } else {
        console.log('Hello ' + name);
    }
})();

8. 坑爹史(1)

var a = [0];
if ([0]) {
  console.log(a == true);
} else {
  console.log("wut");
}

讀者們你們覺得此題答案是什么呢振愿?true?因?yàn)閇0]被看做Boolean是被認(rèn)為是true捷犹,理所當(dāng)然的推出來[0]==true,控制臺(tái)輸出true...看似沒錯(cuò),然而并不是這樣滴~[0]這個(gè)玩意兒在單獨(dú)使用的時(shí)候是被認(rèn)為是true的冕末,但用作比較的時(shí)候它是false...所以正確答案是false萍歉;不信的話,F(xiàn)12控制臺(tái)輸出[0]==false栓霜;看是不是true......

9. 坑爹史(2)

1 + - + + + - + 1

這題應(yīng)該是等同于:(倒著看)

1 + (a)  => 2
a = - (b) => 1
b = + (c) => -1
c = + (d) => -1
d = + (e) => -1
e = + (f) => -1
f = - (g) => -1
g = + 1   => 1

答案是2

10. 坑爹史(3)

function sidEffecting(ary) {
  ary[0] = ary[2];
}
function bar(a,b,c) {
  c = 10
  sidEffecting(arguments);
  return a + b + c;
}
bar(1,1,1)

此題涉及ES6語法翠桦,實(shí)在坑的不行...arguments
首先 The arguments object is an Array-like object corresponding to the arguments passed to a function.也就是說 arguments 是一個(gè) object, c 就是 arguments2, 所以對(duì)于 c 的修改就是對(duì) arguments2 的修改.
所以答案是 21.
然而!!!!!!
當(dāng)函數(shù)參數(shù)涉及到 any rest parameters, any default parameters or any destructured parameters 的時(shí)候, 這個(gè) arguments 就不在是一個(gè) mapped arguments object 了.....請(qǐng)看:

function sidEffecting(ary) {
  ary[0] = ary[2];
}
function bar(a,b,c=3) {
  c = 10
  sidEffecting(arguments);
  return a + b + c;
}
bar(1,1,1)

答案是12...
請(qǐng)讀者細(xì)細(xì)體會(huì)!!

11. 坑爹史(4)

[,,,].join(", ")
[,,,] => [undefined × 3]

因?yàn)閖avascript 在定義數(shù)組的時(shí)候允許最后一個(gè)元素后跟一個(gè),, 所以這是個(gè)長(zhǎng)度為三的稀疏數(shù)組(這是長(zhǎng)度為三, 并沒有 0, 1, 2三個(gè)屬性哦)
答案: ", , "

12. 瀏覽器懵逼史(2)

var a = {class: "Animal", name: 'Fido'};
a.class

這個(gè)題比較流氓.. 因?yàn)槭菫g覽器相關(guān), class是個(gè)保留字(現(xiàn)在是個(gè)關(guān)鍵字了);Fuck!
所以答案不重要, 重要的是自己在取屬性名稱的時(shí)候盡量避免保留字. 如果使用的話請(qǐng)加引號(hào) a['class']

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市胳蛮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌丛晌,老刑警劉巖仅炊,帶你破解...
    沈念sama閱讀 217,907評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異澎蛛,居然都是意外死亡抚垄,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門谋逻,熙熙樓的掌柜王于貴愁眉苦臉地迎上來呆馁,“玉大人,你說我怎么就攤上這事毁兆≌懵耍” “怎么了?”我有些...
    開封第一講書人閱讀 164,298評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵气堕,是天一觀的道長(zhǎng)纺腊。 經(jīng)常有香客問我,道長(zhǎng)茎芭,這世上最難降的妖魔是什么揖膜? 我笑而不...
    開封第一講書人閱讀 58,586評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮梅桩,結(jié)果婚禮上壹粟,老公的妹妹穿的比我還像新娘。我一直安慰自己宿百,他們只是感情好趁仙,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,633評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著犀呼,像睡著了一般幸撕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上外臂,一...
    開封第一講書人閱讀 51,488評(píng)論 1 302
  • 那天坐儿,我揣著相機(jī)與錄音,去河邊找鬼。 笑死貌矿,一個(gè)胖子當(dāng)著我的面吹牛炭菌,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播逛漫,決...
    沈念sama閱讀 40,275評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼黑低,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了酌毡?” 一聲冷哼從身側(cè)響起克握,我...
    開封第一講書人閱讀 39,176評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎枷踏,沒想到半個(gè)月后菩暗,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,619評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡旭蠕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,819評(píng)論 3 336
  • 正文 我和宋清朗相戀三年停团,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片掏熬。...
    茶點(diǎn)故事閱讀 39,932評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡佑稠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出旗芬,到底是詐尸還是另有隱情舌胶,我是刑警寧澤,帶...
    沈念sama閱讀 35,655評(píng)論 5 346
  • 正文 年R本政府宣布岗屏,位于F島的核電站辆琅,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏这刷。R本人自食惡果不足惜婉烟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,265評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望暇屋。 院中可真熱鬧似袁,春花似錦、人聲如沸咐刨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,871評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽定鸟。三九已至而涉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間联予,已是汗流浹背啼县。 一陣腳步聲響...
    開封第一講書人閱讀 32,994評(píng)論 1 269
  • 我被黑心中介騙來泰國打工材原, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人季眷。 一個(gè)月前我還...
    沈念sama閱讀 48,095評(píng)論 3 370
  • 正文 我出身青樓余蟹,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親子刮。 傳聞我的和親對(duì)象是個(gè)殘疾皇子威酒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,884評(píng)論 2 354

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

  • 三昔字、閉包和高階函數(shù) 3.1 閉包 3.1.1 變量的作用域 所謂變量的作用域些阅,就是變量的有效范圍旨巷。通過作用域的劃分...
    梁同學(xué)de自言自語閱讀 1,452評(píng)論 0 6
  • FreeCodeCamp - Basic JavaScript 寫在前面: 我曾經(jīng)在進(jìn)谷前刷過這一套題把沼,不過當(dāng)時(shí)只...
    付林恒閱讀 16,443評(píng)論 5 28
  • 第5章 引用類型(返回首頁) 本章內(nèi)容 使用對(duì)象 創(chuàng)建并操作數(shù)組 理解基本的JavaScript類型 使用基本類型...
    大學(xué)一百閱讀 3,233評(píng)論 0 4
  • @轉(zhuǎn)自GitHub 介紹js的基本數(shù)據(jù)類型。Undefined萨咕、Null谍憔、Boolean跺撼、Number病线、Strin...
    YT_Zou閱讀 1,158評(píng)論 0 0
  • 在恍惚與忙碌之間,我已大四鲤嫡。 選擇在一個(gè)這樣的雨夜寫下這篇文字送挑,只是因?yàn)橛暌贵@醒了內(nèi)心的那一絲恐懼感。從未敢對(duì)生活...
    老郎閱讀 349評(píng)論 0 2