這期內容當中小編將會給大家帶來有關怎么在JavaScript中實現new,文章內容豐富且以專業的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
一般來說在js中大概是這樣的:
function Mynew(parent,...rest){ let obj={}; obj.__proto__=parent.prototype; let res=parent.apply(obj,rest); return typeof res=="object"?res:obj; }
上面提到的“空對象”只是簡稱。事實上,就算是定義一個
{}
,它也會從Object.prototype
上繼承很多方法和屬性。
《JavaScript語言精髓與編程實踐》中提出了一個“更加空白的對象”:它有兩種情況 ——Constructor.prototype
值為null;或者Object.getPrototypeOf(obj)
值為null。
但通常我們還能見到另一種寫法:
function Mynew_2(parent,...rest){ let child=Object.create(parent.prototype); let result=parent.apply(child,rest); return typeof result=="object"?result:child; }
這兩種實現的不同無疑引起了我的興趣!
我們來看一段代碼:
var Best1=function(){ this.a=2; } var o1=Object.create(Best1); var o2=Object.create(Best1.prototype); console.log(o1.a); // undefined console.log(o2.a); // undefined
可以看到,以 o1 為例,Object.create()
失去了對原來對象屬性的訪問,而 o2 …同理。
再來看另一段代碼:
var Best=function(){ this.a=2; } Best.prototype.a=3; var o1=Object.create(Best); var o2=Object.create(Best.prototype); console.log(o1.a); // undefined console.log(o2.a); // 3
到這里,我似乎明白了什么,又似乎沒明白…
我決定這么做:
let b=new Best(); console.log(b);
console.log(Best.prototype);
恍然大悟!
原來我們平時說的 “當讀取實例屬性時,如果找不到,就會查找與對象關聯的原型中的屬性;如果還找不到,就去找原型的原型,直到最頂層(__proto__
為null)為止” 是指 “一直順著 __proto__
向上查找”(注意:不經過prototype?。?。
! 那 __proto__
和 prototype
之間又有什么呢?
console.log(b.__proto__); console.log(Best.prototype);
哦!這就是我們常聽到的“實例的 __proto__
等于對象的 prototype
”吧。
那現在回過頭來,打印一下 o2 這個對象:
console.log(o2);
你有沒有想到什么?
根據上面所描述的那樣,o2 就是 Best 的實例??!
所以說,create()
函數實際上返回了一個對象的實例?
但 o1 仍然“不為所動”!
所以我們能否猜測:在 Object.create()
函數中應該是拿到對象的原型并以實例的形式返回:
Object.create=function(o){ let F=function(){}; F.prototype=o; return new F(); }
為什么這里要用函數再 new 的方式?而不是直接用對象去接收?
這和“為什么vue中data是一個函數而不是直接的對象”其實是一個問題:JavaScript中的對象是引用類型,在一個實例中改變某一個元素的值其余實例的值都會發生改變!
而通過create函數則不會:
JS是JavaScript的簡稱,它是一種直譯式的腳本語言,其解釋器被稱為JavaScript引擎,是瀏覽器的一部分,主要用于web的開發,可以給網站添加各種各樣的動態效果,讓網頁更加美觀。
上述就是小編為大家分享的怎么在JavaScript中實現new了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關知識,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。