成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

【芝士整理】JS基礎(chǔ)圖譜

netScorpion / 2444人閱讀

摘要:沒(méi)有找到的話,看上級(jí)函數(shù)作用域,向上查找到,找到為止。將會(huì)在執(zhí)行上下文棧中保留上級(jí)作用域的執(zhí)行上下文。若在閉包使用完畢之后不手動(dòng)解除引用,相關(guān)執(zhí)行上下文將會(huì)一直保留于執(zhí)行上下文棧中,占據(jù)內(nèi)存空間,若持續(xù)積累,容易造成內(nèi)存泄漏。

JS有哪些基本數(shù)據(jù)類型呢?

值類型:undefined, Number, Boolean, String,null

引用類型:Object

值類型存放在棧中

引用類型將地址存放在棧中,將數(shù)據(jù)實(shí)體存放在堆中

null和undefined,not defined的區(qū)別?

not defined是未聲明,當(dāng)使用未聲明變量時(shí)瀏覽器會(huì)拋出這個(gè)錯(cuò)誤

undefined是已聲明未賦值,typeof undefined是undefined

null類似于空對(duì)象,是一個(gè)已定義,定義為空的值,typeof null 是 object

如何判斷數(shù)據(jù)類型?

如果是值類型,直接用typeof判斷

如果是引用類型,使用instanceof判斷,instanceof基于原型鏈,一般用于判斷自定義對(duì)象

constructor是prototype上的一個(gè)屬性,他容易被重寫覆蓋,所以不可信賴

Object.prototype.toString.call,調(diào)用Object原型上的toString方法可以得到當(dāng)前調(diào)用者的具體類型

Object.prototype.toString.call().slice(8, -1); // Object|Array|Number|String|Boolean...
什么是原型鏈?

每一個(gè)函數(shù)上都有一個(gè)prototype屬性,稱為原型對(duì)象

函數(shù)實(shí)例化產(chǎn)生對(duì)象

每一個(gè)對(duì)象都有一個(gè)__proto__(隱匿原型)屬性,指向構(gòu)造它的原型對(duì)象。

原型對(duì)象本身也是對(duì)象,也有一個(gè)隱匿原型,指向它的原型對(duì)象。

沿著隱匿原型鏈最終會(huì)指向Object.prototype,它的原型對(duì)象是null

這就構(gòu)成一個(gè)原型鏈

PS. 將原子類型賦給 prototype 的操作將會(huì)被忽略

function Foo() {}
Foo.prototype = 1; // 無(wú)效

instanceof的原理

A instanceof B
A的原型鏈?zhǔn)欠駮?huì)到達(dá)B.prototype

繼承

通過(guò)原型鏈實(shí)現(xiàn)繼承,原型對(duì)象上可以定義屬性和方法。

當(dāng)要在一個(gè)對(duì)象上尋找某個(gè)屬性,先在對(duì)象本身找,沒(méi)有的話,再沿著原型鏈向上找原型對(duì)象里有沒(méi)有,向上查找找到為止,到達(dá)頂部仍未找到,返回undefined

PS.判斷對(duì)象上是否有某個(gè)屬性,而非其原型鏈上有,使用hasOwnProperty 函數(shù)
執(zhí)行上下文

在函數(shù)調(diào)用時(shí)或者是全局代碼開始運(yùn)行時(shí)產(chǎn)生,處理的事情:變量聲明,函數(shù)聲明,函數(shù)聲明形式的定義賦值,定義this,在函數(shù)內(nèi)還有定義arguments的操作

PS. arguments 變量不是一個(gè)數(shù)組(Array)。 盡管在語(yǔ)法上它有數(shù)組相關(guān)的屬性 length,但它不從 Array.prototype 繼承,實(shí)際上它是一個(gè)對(duì)象(Object)。

因此,無(wú)法對(duì) arguments 變量使用標(biāo)準(zhǔn)的數(shù)組方法,比如 push, pop 或者 slice。 雖然使用 for 循環(huán)遍歷也是可以的,但是為了更好的使用數(shù)組方法,最好把它轉(zhuǎn)化為一個(gè)真正的數(shù)組。

Array.prototype.slice.call(arguments);
執(zhí)行上下文棧

全局代碼開始執(zhí)行時(shí),產(chǎn)生一個(gè)全局的執(zhí)行上下文,壓棧

代碼執(zhí)行到函數(shù)A調(diào)用時(shí),產(chǎn)生一個(gè)函數(shù)A的執(zhí)行上下文,壓棧

函數(shù)A中調(diào)用函數(shù)B,產(chǎn)生一個(gè)函數(shù)B的執(zhí)行上下文,壓棧

函數(shù)B,執(zhí)行完畢,出棧銷毀執(zhí)行上下文

函數(shù)A,執(zhí)行完畢,出棧并銷毀執(zhí)行上下文

關(guān)于this的指向

this存在于執(zhí)行上下文中

作為構(gòu)造函數(shù)調(diào)用時(shí),指向?qū)嵗膶?duì)象

作為對(duì)象屬性調(diào)用時(shí),指向?qū)ο?/p>

call, apply調(diào)用時(shí),指向參數(shù)指定上下文

全局和普通函數(shù)都指向windows

ES6中,箭頭函數(shù)本身沒(méi)有this,導(dǎo)致以下三種現(xiàn)象:根據(jù)外層(函數(shù)或者全局)作用域來(lái)決定this,箭頭函數(shù)不能作為構(gòu)造函數(shù)使用,不能使用call, apply手動(dòng)修改this

PS. 一些誤解

// 1. 嚴(yán)格按照規(guī)范
Foo.method = function() {
    // 在這,this是Foo的實(shí)例化對(duì)象
    function test() {
        // this 將會(huì)被設(shè)置為全局對(duì)象
    }
    test();
}

// 2. 函數(shù)別名
var test = someObject.methodTest;
test(); // this設(shè)置為全局對(duì)象
PS. apply和call的用法

function.apply(null, arguments);

function.call(null, arg1, arg2);

作用域

ES5中只有函數(shù)作用域的概念,作用域是一個(gè)虛擬概念,沒(méi)有具體的數(shù)據(jù)類型或者結(jié)構(gòu)。

一個(gè)函數(shù)的作用域在函數(shù)定義時(shí)確定,創(chuàng)建函數(shù)的作用域成為該函數(shù)的上級(jí)作用域

在函數(shù)中尋找變量,先找到函數(shù)作用域?qū)?yīng)的執(zhí)行上下文,在執(zhí)行上下文中找變量。

沒(méi)有找到的話,看上級(jí)函數(shù)作用域,向上查找到,找到為止。

如果找不到,則會(huì)拋出 ReferenceError異常。

PS. 比如,當(dāng)訪問(wèn)函數(shù)內(nèi)的 foo 變量時(shí),JavaScript 會(huì)按照下面順序查找:

當(dāng)前作用域內(nèi)是否有 var foo 的定義。

函數(shù)形式參數(shù)是否有使用 foo 名稱的。

函數(shù)自身是否叫做 foo

回溯到上一級(jí)作用域,然后從 #1 重新開始。

閉包

什么是閉包?
一個(gè)函數(shù)中有依賴外部變量,函數(shù)在創(chuàng)建它的作用域之外被調(diào)用。
將會(huì)在執(zhí)行上下文棧中保留上級(jí)作用域的執(zhí)行上下文。
若在閉包使用完畢之后不手動(dòng)解除引用,相關(guān)執(zhí)行上下文將會(huì)一直保留于執(zhí)行上下文棧中,占據(jù)內(nèi)存空間,若持續(xù)積累,容易造成內(nèi)存泄漏。

常見應(yīng)用,函數(shù)作為返回值,函數(shù)作為參數(shù)。

經(jīng)典問(wèn)題

for (var i = 0; i < 5; i++) {
    setTimeout(function() {  
      console.log(i);
      }, 1000);
} // 5,5,5,5,5

for (let i = 0; i < 5; i++) {
    setTimeout(function() {  
      console.log(i);
      }, 1000);
} // 0,1,2,3,4

for (var i = 0; i < 5; i++) {
    (function (i) {
        setTimeout(function() {  
          console.log(i);
          }, 1000);
    })(i)
} // 0, 1, 2, 3, 4

for(var i = 0; i < 5; i++) {
    setTimeout((function(e) {
        return function() {
            console.log(e);
        }
    })(i), 1000)
} // 0, 1, 2, 3, 4
JS中實(shí)現(xiàn)繼承的方法

基礎(chǔ)繼承

var Bar = function () {};
Bar.prototype = {
    greet: function (name) {
        console.log(name);
    }
}
var Foo = function () {}
Foo.prototype.__proto__ = Bar.prototype;

等價(jià)于

var Bar = function () {};
var Foo = function () {}
Foo.prototype = new Bar();

原理:實(shí)現(xiàn)原型鏈

缺點(diǎn):屬性不獨(dú)立

組合繼承

var Bar = function (name) {
    this.name = name;
}
Bar.prototype = {
    greet: function () {
        console.log(this.name);
    }
}
var Foo = function (name) {
    Bar.apply(this, arguments);
}
Foo.prototype = new Bar();

原理:把this屬性賦值在子類的作用域執(zhí)行一次,方法通過(guò)原型鏈繼承
缺點(diǎn):this屬性賦值進(jìn)行了兩次

寄生組合式繼承

var Bar = function (name) {
    this.name = name;
}
Bar.prototype = {
    greet: function () {
        console.log(this.name);
    }
}
var Foo = function (name) {
    Bar.apply(this, arguments);
}
Foo.prototype = Object.create(Bar.prototype);
Foo.prototype.constructor = Foo;

原理: 把this屬性賦值在子類的作用域執(zhí)行一次,手動(dòng)連接原型對(duì)象的拷貝
優(yōu)點(diǎn):解決組合繼承的缺點(diǎn)

extends方法

class Point {
    constructor(x, y) {
        this.x = x;
        this.y = y;
    }
    
    toString () {
        return this.x + " " + this.y;
    }
}

class ColorPoint extends Point {
  constructor(x, y, color) {
    super(x, y); // 調(diào)用父類的constructor(x, y)
    this.color = color;
  }

  toString() {
    return this.color + " " + super.toString(); // 調(diào)用父類的toString()
  }
}

子類自己的this對(duì)象,必須先通過(guò)父類的構(gòu)造函數(shù)完成塑造,得到與父類同樣的實(shí)例屬性和方法,然后再對(duì)其進(jìn)行加工,加上子類自己的實(shí)例屬性和方法。如果不調(diào)用super方法,子類就得不到this對(duì)象。

PS. new 運(yùn)算符做了什么

// 1. 首先創(chuàng)建一個(gè)空對(duì)象
var o = new Object();
// 2. 將空對(duì)象的原型賦值為構(gòu)造器函數(shù)的原型
o.__proto__ = A.prototype;
// 3. 更改構(gòu)造器函數(shù)內(nèi)部this,將其指向新創(chuàng)建的空對(duì)象
A.call(o);
類型轉(zhuǎn)換 強(qiáng)制轉(zhuǎn)換

轉(zhuǎn)為字符串::.toString(), String()

轉(zhuǎn)為數(shù)值:Number針對(duì)所有類型,parseInt和parseFloat針對(duì)字符串

字符串轉(zhuǎn)換為數(shù)字的常用方法:

+"010" === 10
Number("010") === 10
parseInt("010", 10) === 10  // 用來(lái)轉(zhuǎn)換為整數(shù)

+"010.2" === 10.2
Number("010.2") === 10.2
parseInt("010.2", 10) === 10
parseFloat("10.1.2") === 10.1 // 字符轉(zhuǎn)換為浮點(diǎn)數(shù)

Number(undefined) // NaN
Number嚴(yán)格轉(zhuǎn)換,只要有一個(gè)字符無(wú)法轉(zhuǎn)為數(shù)值輸出NaN
parseInt原理為從左往右讀字符串,讀到非數(shù)值字符為止
parseFloat原理為從左往右讀字符串,讀到第二個(gè)小數(shù)點(diǎn)或者非數(shù)值非小數(shù)點(diǎn)字符為止

轉(zhuǎn)為布爾值:Boolean(),""、null、undefined、+0、-0 和 NaN 轉(zhuǎn)為布爾型是 false,其他的都是 true

自動(dòng)轉(zhuǎn)換

轉(zhuǎn)為字符串:包含字符串的+法運(yùn)算,"5" + 1 === "51"

轉(zhuǎn)為數(shù)值:不包含字符串的算術(shù)運(yùn)算

轉(zhuǎn)為布爾值:條件語(yǔ)句判斷,非運(yùn)算

事件輪詢機(jī)制

主線程運(yùn)行時(shí)產(chǎn)生堆和執(zhí)行棧

主線程之外,還存在一個(gè)"任務(wù)隊(duì)列"。只要異步任務(wù)有了運(yùn)行結(jié)果,就在"任務(wù)隊(duì)列"之中放置一個(gè)事件。

一旦"執(zhí)行棧"中的所有同步任務(wù)執(zhí)行完畢,系統(tǒng)就會(huì)讀取"任務(wù)隊(duì)列",看看里面有哪些事件。對(duì)應(yīng)的異步任務(wù),結(jié)束等待狀態(tài),進(jìn)入執(zhí)行棧,開始執(zhí)行

任務(wù)隊(duì)列

任務(wù)隊(duì)列分為宏任務(wù)隊(duì)列和微任務(wù)隊(duì)列

宏任務(wù)包括:script(全局任務(wù)), setTimeout, setInterval, setImmediate, I/O, UI rendering。

微任務(wù)包括: new Promise().then(回調(diào)), process.nextTick, Object.observe(已廢棄), MutationObserver(html5新特性)

執(zhí)行同步任務(wù) -> 處理微任務(wù)隊(duì)列 -> 處理宏任務(wù)隊(duì)列里隊(duì)首任務(wù) -> 處理微任務(wù)隊(duì)列

Promise.resolve().then(()=>{
  console.log("Promise1")  
  setTimeout(()=>{
    console.log("setTimeout1")
  },0)
})
setTimeout(()=>{
  console.log("setTimeout2")
  Promise.resolve().then(()=>{
    console.log("Promise2")    
  })
  Promise.resolve().then(()=>{
  console.log("Promise3")    
  })
},0)
setTimeout(()=>{
  console.log("setTimeout4")
  Promise.resolve().then(()=>{
    console.log("Promise4")    
  })
},0)

Output: Promise1 setTimout2 Promise2 Promise3 setTimeout4 Promise4 setTimeout1
setTimeout和setInterval的區(qū)別

setTimeout:產(chǎn)生一個(gè)宏任務(wù),在指定時(shí)間之后加入任務(wù)隊(duì)列。

setInterval:循環(huán)產(chǎn)生宏任務(wù),但存在問(wèn)題,若任務(wù)執(zhí)行時(shí)間長(zhǎng)于指定時(shí)間間隔,會(huì)產(chǎn)生堆疊執(zhí)行效果。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/100410.html

相關(guān)文章

  • 芝士整理】CSS基礎(chǔ)圖譜

    摘要:為了實(shí)現(xiàn)文字環(huán)繞效果,規(guī)范規(guī)定的措施是使父容器塌陷,元素脫離文檔流浮動(dòng)產(chǎn)生,元素周圍的內(nèi)容轉(zhuǎn)換為圍繞元素排列。 選擇器注意點(diǎn) 屬性選擇器 [attr^=value] - 開頭或全等 [attr$=value] - 結(jié)尾或全等 [attr*=value] - 包含值 [attr~=value] - 字符串包含 選擇器組 A > B - 直接子節(jié)點(diǎn) A + B - 下一個(gè)兄弟節(jié)點(diǎn) A...

    iOS122 評(píng)論0 收藏0
  • 芝士整理】HTML的標(biāo)簽們

    摘要:讀一遍文檔后的個(gè)人總結(jié),重點(diǎn)在于整理語(yǔ)義化標(biāo)簽的定義規(guī)范,記錄各種部件容易被忽略的特性。結(jié)構(gòu)化,通過(guò)標(biāo)簽先后順序和嵌套語(yǔ)法給樹提供基礎(chǔ)。標(biāo)簽列表基于個(gè)人理解即非官方描述,給標(biāo)簽劃分為結(jié)構(gòu)化標(biāo)簽語(yǔ)義化標(biāo)簽功能化標(biāo)簽,文檔標(biāo)簽。 讀一遍MDN文檔后的個(gè)人總結(jié),重點(diǎn)在于整理語(yǔ)義化標(biāo)簽的定義規(guī)范,記錄各種部件容易被忽略的特性。 關(guān)于HTML HTML的作用可以簡(jiǎn)單總結(jié)為結(jié)構(gòu)化、語(yǔ)義化和提供基礎(chǔ)...

    stonezhu 評(píng)論0 收藏0
  • 【前端芝士樹】Array的屬性及方法整理(參照MDN)

    摘要:本文主要是我自己對(duì)的一些整理,參考自,其中的分類有些不準(zhǔn)確之處,還望見諒的基本屬性屬性的一些方法增刪改查基礎(chǔ)功能增刪改查基礎(chǔ)功能增刪改刪除數(shù)組的第一個(gè)元素刪除數(shù)組的最后一個(gè)元素在數(shù)組的開頭一個(gè)或多個(gè)元素,在數(shù)組的末尾增加一個(gè)或者多個(gè)元素?cái)?shù)組 本文主要是我自己對(duì)Array的一些整理,參考自MDN,其中的分類有些不準(zhǔn)確之處,還望見諒 Array const arr = [1, 2, 3, ...

    MoAir 評(píng)論0 收藏0
  • 芝士整理】瀏覽器存儲(chǔ)

    摘要:維護(hù)瀏覽器和服務(wù)器端會(huì)話狀態(tài)的一種方式,一般用于保存用戶身份信息。服務(wù)器端生成推送到瀏覽器端,瀏覽器負(fù)責(zé)保存和維護(hù)數(shù)據(jù)。 Cookie 維護(hù)瀏覽器和服務(wù)器端會(huì)話狀態(tài)的一種方式,一般用于保存用戶身份信息。 服務(wù)器端生成Cookie推送到瀏覽器端,瀏覽器負(fù)責(zé)保存和維護(hù)數(shù)據(jù)。 特點(diǎn) 域名下的所用請(qǐng)求都會(huì)帶上Cookie 每條Cookie限制在4KB左右 Cookie在過(guò)期時(shí)間之前一直有效,若...

    whlong 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<