国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

JS面向對象二:this/原型鏈/new原理

anRui / 1274人閱讀

摘要:情況沒有明確作用對象的情況下,通常為全局對象例如函數的回調函數,它的就是全局對象。正因如此,機器可以作為這類對象的標志,即面向對象語言中類的概念。所以機器又被稱為構造函數。原型鏈也就是繼承鏈。

JS面向對象二:this/原型鏈/new原理

阮一峰JavaScript教程:面向對象編程

阮一峰JavaScript教程:實例對象與 new 命令

阮一峰JavaScript教程:this 關鍵字

也可以看看這篇文章周大俠啊 進擊的 JavaScript(六) 之 this先了解一下`this的四種綁定規則和箭頭函數的this綁定

this

這兩篇文章寫的很好
周大俠啊 進擊的 JavaScript(六) 之 this

蘇云的博客

下面的this分別是什么?這幾個函數都是回調函數,回調函數this比較特殊,通常是事件原對象

//1.
button1.onclick = function () {
    console.log(this)
}
//2.
button2.addEventListener("click", function () {
    console.log(this)
})
//3.jQuery中
$("ul").on("click", "li", function () {
    console.log(this)
})

this是函數.call()的第一個參數.

那么在直接調用函數的時候(隱式綁定,沒用call),如何知道call()的第一個參數?
源碼看不到,那就看文檔.

看文檔!:
onclick:

addEventListener:

jQuery中:

所以上面三個的this分別是
btutton1元素,button2元素,li元素

$("ul").on("click", "li"/*selector*/, function () {
console.log(this)//代表與selector相匹配的元素(li元素)
})

thiscall()的第一個參數,只有寫onclick,寫addEventListener和寫jQueryon的人想call()哪個東西,就把這個this綁定到哪里去了,所以要確定this,就要看源碼或者文檔!

例如:

button1.onclick = function () {
    console.log(this)
}
button1.onclick.call({name:"mataotao"})

可以直接觸發onclick事件,傳入{name:"mataotao"},那么this就是{name:"mataotao"}這個對象

以下來自蘇云的博客()

6.回調函數的this 回調函數也只不過是函數的一種,實際上這種情況已經包含在了前面提到的情況中。但是由于回調函數的調用者往往不是我們自己,而是回調函數的接收者,即某個庫或框架、甚至是JS運行時環境。這樣一來,回調函數在中的this是什么就與對方的調用方式有關了,因此變得比較復雜,所以多帶帶拿出來討論一下。

情況1:沒有明確作用對象的情況下,通常this為全局對象

例如setTimeout函數的回調函數,它的this就是全局對象。你如果希望自己指定this,可以通過bind函數等方法。

情況2:某個事件的監聽器回調函數,通常this就是事件源對象

例如:

button.addEventListener("click", fn)

fn的中的this就是事件源button對象。

情況3:某些API會專門提供一個參數,用來指定回調函數中的this

例如,我們可以重新設計一個可以指定thissetTimeout

function setTimeoutExt(cb, period, thisArg) {
    setTimeout(function() {
        cb.call(thisArg);
    }, period); }

另外,在ExtJS中也大量使用了可以指定this的接口。

this題目

答案:
調用B處的console.log().結果是options
window(console.log()中console是全局window對象里的一個方法)

第二題:

答案:D Object

第三題:

答案:Object

原型鏈

我終于明白了原型鏈:
仔細看下面這篇文章,就能明白原型鏈的構造問題:
JavaScript 世界萬物誕生記

個人理解:
原型鏈要分為兩個部分,原型和鏈,原型就是一個實例對象,但是是最基礎的實例對象.這個實例對象可以作為模板/類,讓其他對象去復制他,復制之后不單單有這個原型的屬性,也可以有自己的屬性.新實現的實例對象.__proto__指向原來的模板實例對象.
而造出來的對象也可以當做模板,再由新的機器去以他為模板造新對象.由此形成了一條__proto__組成的鏈.
所有的對象都有__proto__屬性,他們就像被鏈子連接在了一起,所以就稱之為原型鏈

而復制的過程由一個機器來完成.這個機器(比如可以說是Object())的使用方法就是:按照模板實例對象new()一個新對象,新對象被原來的模板對象用__proto__鏈子拴著,新對象可以有自己的新添加的東西.
這個按照模板造新對象的機器.prototype指向原來的模板實例對象.prototype就是原來的模板實例對象拴住復制自己機器的鏈子.

寫成代碼就是:

var obj = new Object({ flag: 10 });
就像前面所說,機器用來制造某一類對象。正因如此,機器可以作為這類對象的標志,即面向對象語言中(class)的概念。所以機器又被稱為構造函數。在ES6引入class關鍵字之前,我們常常把構造函數叫做類。

說明2:用戶自定義的函數通常既可以作為普通函數使用,又可以作為構造函數來制造對象。ES6新增的class語法定義的函數只能作為構造函數,ES6新增的=>語法定義的箭頭函數只能作為普通函數

.

__proto__prototype的區別
__proto__是所有對象(包括函數對象)都有的一個屬性(當然只是邏輯上有這么個概念),當我們說“原型鏈”的時候,就是指對象通過這個屬性互相連接而形成的鏈狀結構原型鏈也就是繼承鏈
prototype是只有函數(準確地說是構造函數)才有的一個屬性,例如對于對于某個函數Fun。它的意義在于,當你用var obj = new
Fun() 得到一個對象obj時,這個obj的原型就是F.prototype。即(new Fun()).__proto__ ===Fun.prototype,見文中第4個圖。

No. 1其實就是Object.prototype,No.
2其實就是Function.prototype。我只是為了強調這兩個對象的重要性,故意這樣說的。不太嚴格地說,前者就是一個空對象類似:{}
,后者就是一個空函數,類似:fubction() {} 。

文中:
**No. 1:Object.prototype
No. 2:Function.prototype**

還有這幾篇文章也不錯:
「每日一題」什么是 JS 原型鏈? - 方應杭的文章 - 知

周大俠啊 進擊的 JavaScript 之 (七) 原型鏈

周大俠啊 進擊的 JavaScript (八) 之 繼承

new()

看看這篇文章很清楚:
JS 的 new 到底是干什么的? - 方應杭的文章 - 知乎

new解決了什么

以共有屬性對象為模板new出來的新對象的__proto__指向共有屬性對象(我把這個對象叫做模板對象,也叫作原型).這樣共有屬性在內存中只需要存一次!

比如:當我們造士兵的時候,士兵有共有屬性,有自有屬性,那么我們可以把共有屬性放在一個地方,避免每一次創建士兵都把共有屬性重新創建一次,浪費內存:

既然這樣,那么我們可以把制造士兵的過程寫成一個函數.

然后調用即可

直接使用函數就可以制造一個有特殊的id,但是__proto__指向原型士兵的新士兵

那么可不可以直接把這個原型對象放到函數里,組成一個整體?不行,這樣每次調用這個函數,都會在內存中創建這個臨時對象,那么和原先的不用原型一樣了

解決方法是,把這個原型變為函數的一個屬性

這種方法省內存且好用.

new()就是剛剛的所有過程

灰色的代碼就是new()做的封裝,不需要你做的事情
共有屬性被new()統一叫做prototype

new其實就是語法糖!

注意:.prototype對象最開始就是一個擁有constructor屬性的對象,如果想修改共有屬性,兩種方法:

當我們new的時候我們做了什么:

new()的核心就是:

new應用舉例:

第一步寫私有屬性,第二步寫共有屬性.

可以看到這個對象的
1自有屬性,
2__proto__指向的原型對象含有共有屬性.
3constructor指向的構造函數

節選文章

也可以看看這篇文章周大俠啊 進擊的 JavaScript(六) 之 this
里面有new的實現.new與this
下面是節選:

五、new 綁定 如果 使用 new 來創建對象,因為
后面跟著的是構造函數,所以稱它為構造器調用。對于this綁定來說,稱為new綁定。

想知道 構造器調用 中 this 的綁定,就要知道 new 到底做了啥了。

先來個 new 的實現。看不懂不要緊,在后面原型鏈那篇,還會說的。

function New(proto){  //proto 為傳進來的構造函數
    var obj = {};
    obj.__proto__ = proto.prototype;

    proto.apply(obj, Array.prototype.slice.call(argument,1));
    //你這要看懂這步就行。這里把構造函數里的 this  綁定到了 新的obj 對象上,最后 返回了該新對象,作為實例對象。

    return obj; }

所以在使用 new 來創建實例對象時,new 內部把 構造函數的 this 綁定到 返回的新對象 上了。

function Person(name){
    this.name = name; } var c = new Person("zdx"); c.name;

JavaScript 世界萬物誕生記中也提到了new的使用,new的過程就是生產機器按照模板原型對象造出來的新對象的過程!

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/108408.html

相關文章

  • 前端進擊的巨人(七):走進面向對象原型原型,繼承方式

    摘要:除了以上介紹的幾種對象創建方式,此外還有寄生構造函數模式穩妥構造函數模式。 showImg(https://segmentfault.com/img/remote/1460000018196128); 面向對象 是以 對象 為中心的編程思想,它的思維方式是構造。 面向對象 編程的三大特點:封裝、繼承、多態: 封裝:屬性方法的抽象 繼承:一個類繼承(復制)另一個類的屬性/方法 多態:方...

    wums 評論0 收藏0
  • 面向對象的 JavaScript

    摘要:是完全的面向對象語言,它們通過類的形式組織函數和變量,使之不能脫離對象存在。而在基于原型的面向對象方式中,對象則是依靠構造器利用原型構造出來的。 JavaScript 函數式腳本語言特性以及其看似隨意的編寫風格,導致長期以來人們對這一門語言的誤解,即認為 JavaScript 不是一門面向對象的語言,或者只是部分具備一些面向對象的特征。本文將回歸面向對象本意,從對語言感悟的角度闡述為什...

    novo 評論0 收藏0
  • 我來重新學習js面向對象(part 4)

    摘要:我是的可以改變函數的對象的指向拋出異常,沒有這個因為子類和超類都是構造函數,那么就會有之前說的,構造函數在的時候,里面的方法函數會重復創建實例,導致資源浪費。 我來重新學習js 的面向對象(part 4) 續上一篇,隨著業務越來越大,要考慮一些繼承的玩意了,大千世界,各種東西我們要認識和甄別是需要靠大智慧去分門別類,生物學中把動植物按界、門、綱、目、科、屬、種進行分類的方法可能是最有代...

    MAX_zuo 評論0 收藏0
  • js面向對象淺談(三)

    摘要:還有一個問題,就是不能在創建子類性時,像父類型的構造函數傳遞參數。組合繼承將原型鏈和借用構造函數組合到一起,發揮兩者之長的一張繼承模式,下面來看個例子。組合繼承最大的問題是無論在什么情況下,都會調用兩次父類型構造函數。 繼承 繼承是面向對象語言中特別重要的概念,js的繼承主要是靠原型鏈實現的。 原型鏈!!! 看到我給標題打了三個嘆號嗎,這里真的很重要!這里真的很重要!這里真的很重要!j...

    awkj 評論0 收藏0
  • 深入理解 JavaScript 中的 class

    摘要:在規范中,引入了的概念。使用中的聲明一個類,是非常簡單的事。中面向對象實例化的背后原理,實際上就是原型對象。與區別理解上述原理后,還需要注意與屬性的區別。實際上,在中,類繼承的本質依舊是原型對象。 在 ES6 規范中,引入了 class 的概念。使得 JS 開發者終于告別了,直接使用原型對象模仿面向對象中的類和類繼承時代。 但是JS 中并沒有一個真正的 class 原始類型, clas...

    Vicky 評論0 收藏0

發表評論

0條評論

anRui

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<