摘要:請看下方的例子我們可以看一下構造函數的原型對象再來看一下的原型對象指針指向的對象所以在修改了原型對象之后,只有在修改之后生成的實例對象上會有新的屬性和方法,之前的實例對象指向的仍然是之前的原型對象。
引出問題
function Person(name, age) {} // Person構造函數是個空函數,將默認值都放在原型對象中 Person.prototype.name = "xiaoming" Person.prototype.age = 20 Person.prototype.sayHi = function() { console.log(`hi, ${this.name}`) } const p1 = new Person() /* 實例對象的內部指針指向的是構造函數的原型對象 構造函數的prototype屬性指向的也是這個原型對象 實例對象和構造函數之間沒有直接的聯系 */ console.log(p1.__proto__ === Person.prototype) //true console.log(Person.prototype.constructor) /* ? Person(name, age) { this.name = name this.age = age this.sayHi = function() { console.log(`hi, ${this.name}`) } 構造函數的原型對象中的constructor指向的是構造函數 } */ // 嘗試重寫Person構造函數的原型對象 Person.prototype = { name: "alice", age: 12, sayLove: function() { console.log(`i love ${this.name}`) } } console.log(Person.prototype.constructor) // ? Object() { [native code] }
重寫了 prototype 之后發現它的constructor不再指向 Person,而是指向了Object構造函數
why ?
明確一點,在調用構造函數的時候,會為生成的新的實例對象添加一個指針指向構造函數的原型對象
那么在重寫prototype的時候我們用對象字面量的方式創建了一個新的對象,而用這種方式創建就相當于調用了Object構造函數
不信可以試試
const o1 = {} const o2 = new Object() console.log(o1.__proto__ === o2.__proto__) // true
此時,在調用了Object構造函數創建一個新對象,并將這個新對象作為 Person 的 prototype 之后,
發現這個原型對象里面少了一個constructor屬性;
當在一個實例對象中找不到該屬性時,就會去這個實例對象的構造函數的原型對象中尋找,
這個原型對象的構造函數是 Object,所以就會去 Object構造函數的原型對象中尋找,
而我們前面說了,構造函數的原型對象中的constructor指向的是構造函數,
所以Object的原型對象中的constructor指向的還是Object,
那么如何避免這種情況呢?
如果constructor真的很重要,那么在重寫原型對象的時候可以在對象中加上constructor屬性,
這樣的話就不會去新對象的原型對象中查找constructor屬性了
Person.prototype = { constructor: Person, // 注意這個值不是字符串 name: "alice", age: 12, sayLove: function() { console.log(`i love ${this.name}`) } } console.log(Person.prototype.constructor) /* ? Person(name, age) { this.name = name this.age = age this.sayHi = function() { console.log(`hi, ${this.name}`) } } */
仍然需要注意的是,在修改了構造函數的原型對象之后,即使給原型對象添加了constructor屬性,但是之前通過構造函數生成的實例對象不會自動更新它們的原型對象的指針。請看下方的例子:
//我們可以看一下Person構造函數的原型對象 console.log(Person.prototype) // {name: "alice", age: 12, sayLove: ?, constructor: ?} //再來看一下p1的原型對象指針指向的對象 console.log(p1.__proto__) // {name: "xiaoming", age: 20, sayHi: ?, constructor: ?}
所以在修改了原型對象之后,只有在修改之后生成的實例對象上會有新的屬性和方法,之前的實例對象指向的仍然是之前的原型對象。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/99665.html
摘要:我們知道,調用構造函數時會為實例添加一個指向最初原型的指針,而把原型修改為另外一個對象就等于切斷了構造函數與最初原型之間的聯系。 我們知道,調用構造函數時會為實例添加一個指向最初原型的指針,而把原型修改為另外一個對象就等于切斷了構造函數與最初原型之間的聯系。 怎么理解這句話,下面我們通過實例來解釋下,如有不對,請指正,虛心受教。 function Person(){ } ...
摘要:原型對象的共享特性,可以很方便的為一些內置的對象擴展一些方法,比如,數組去重復但是,不要隨便往內置的對象上面擴展方法,在多人協作開發,很容易產生覆蓋,以及污染 我們先來一個簡單的構造函數+原型對象的小程序 1 function CreateObj( uName, uAge ) { 2 this.userName = uName; 3 ...
摘要:先來說其實構造函數也有,原型對象有,實例有也有,或者更加籠統的說,所有對象都是有的。構造函數的原型對象上的會指向構造函數。由于屬性是可以變更的,所以未必真的指向對象的構造函數,只是一個提示。 續上一集內容,通過構造函數的方式,成功地更新了生產技術,老板笑呵呵,工人少奔波,只是問題總比辦法多,又遇到一個新問題,就是會造成一些資源的重復和浪費,那么經過工程師們的智慧交流,他們產生了一個新技...
摘要:中創建對象的方式有很多,尤其是基于原型的方式創建對象,是理解基于原型繼承的基礎。該函數中的屬性指向該源性對象當通過該函數的構造函數創建一個具體對象時,在這個對象中,就會有一個屬性指向原型。 js中創建對象的方式有很多,尤其是基于原型的方式創建對象,是理解基于原型繼承的基礎。因此在這里匯總一下,并對各種方法的利弊進行總結和對比,不至于以后對這些概念有模糊。 簡單方式創建 var o = ...
摘要:創建實例的方式有三種對象字面量表示法操作符跟構造函數中的函數。下面主要講的是最為復雜的操作符跟構造函數的創建對象實例的方法。 創建對象 一.創建對象的方法 理解原型對象: 無論什么時候,只要創建了新函數,就會根據一組特定的規則為該函數創建一個 prototype屬性,這個屬性指向函數的原型對象。在默認情況下,所有原型對象都會自動獲得一個constructor屬性,這個屬性包含一個指向p...
閱讀 655·2021-09-24 09:48
閱讀 2499·2021-08-26 14:14
閱讀 524·2019-08-30 13:08
閱讀 1450·2019-08-29 15:22
閱讀 3084·2019-08-29 11:06
閱讀 1011·2019-08-26 18:26
閱讀 1062·2019-08-26 13:53
閱讀 2538·2019-08-26 12:21