JS:原型与原型链(附带图解与代码) | 您所在的位置:网站首页 › 通讯兵丁小二原型 › JS:原型与原型链(附带图解与代码) |
一、原型
写在前面: 任何对象都有原型。 函数也是对象,所以函数也有原型。 1.什么是原型在 JavaScript 中,对象有一个特殊的隐藏属性 [[Prototype]],它要么为 null,要么就是对另一个对象的引用,该对象被称为“原型”。所以,被 [[Prototype]] 链接所引用的对象就是该对象的“原型”。 我们可以通过Object.getPrototypeOf/Object.setPrototypeOf(推荐写法)或者是__proto__去设置原型,__proto__ 是 [[Prototype]] 的 getter/setter,我们可以用Object.getPrototypeOf/Object.setPrototypeOf 来取代 __proto__ 去 get/set 原型,也可以继续使用__proto__去访问原型。 例如,在下面的示例中,通过Object.create()将对象b的原型设置为a,实现原型式继承,此时对象a就是对象b的原型。 const a = { name: "小王", }; // Object.create()是一个静态方法,它创建一个新对象,使用现有的对象作为新创建的对象的proto。 // 在这个例子中,我们使用对象a作为原型来创建新对象b。这意味着b的原型是a`。 const b = Object.create(a); // Object.getPrototypeOf()方法返回指定对象的原型(即内部[[Prototype]]属性的值)。 console.log(Object.getPrototypeOf(b));//{name: '小王'} // proto是一个非标准的属性,但在许多环境中都可用,它提供了对对象原型的直接访问。这里,b.__proto__也返回a,即{name: '小王'}`。 console.log(b.__proto__);//{name: '小王'} console.log(a); //{name: '小王'} console.log(b.name); //小王图像示例: 直接上图: 原型链图 讲解: 1.构造函数构造函数是一种特殊的函数,通常用于初始化新创建的对象实例。每个构造函数都有一个prototype属性,这个属性是一个指针,指向一个对象(原型对象),该对象的用途是包含可以由特定类型的所有实例共享的属性和方法。当访问一个对象的属性时,如果这个对象自身没有这个属性,JavaScript会查找这个对象的原型(即[[Prototype]]指向的对象),以此类推,直到找到这个属性或者到达原型链的尽头(null)。我们可以使用new关键字来调用一个构造函数,它会创建一个新的空对象,然后将这个新对象的内部链接([[Prototype]])指向构造函数的prototype属性所引用的对象。构造函数的实例和构造函数的原型对象有一个constructor属性指向构造函数。在上面原型链图中,因为对象b继承了实例a,所以b.constructor也指向构造函数A。当然,构造函数也是一个对象,所以它也有原型,通常是Function.prototype。 2.Object.prototype和Function.prototypeObject.prototype是所有JavaScript对象(除了null)的原型。换句话说,几乎所有的对象都会从Object.prototype继承属性和方法,比如.toString(), .hasOwnProperty(), 和 .constructor等。 Function.prototype是所有函数的原型,包括那些被用作构造函数的函数。这意味着所有的函数都会从Function.prototype继承属性和方法,如.apply(), .call(), 和 .bind()等。 所以在查找原型时,在null之前函数会找到Function.prototype,所有对象都会找到Object.prototype。 3.理解原型链原型链是一种实现继承的机制。在上面的原型链图可以看出,通过把一个对象的原型指向另一个对象,可以让这个对象访问另一个对象的属性,最终形成了一个链条一样的结构。在原型链中查找属性或方法时,JavaScript 会从当前对象开始,沿着原型链(即 __proto__ 链)向上查找,直到找到相应的属性或方法或直到到达 Object.prototype 的原型(即 null)。如果找不到,则返回 undefined。 4.代码展示希望下面的代码可以帮你理解上面的原型链图。 function A() { this.name = "a"; } A.prototype.say = function () { return "hello"; }; const a = new A(); const b = Object.create(a); const c = Object.create(b); console.log(c.say()); //hello,因为c.__proto__指向b,b.__proto__指向a,所以c可以访问到a的原型链上的方法 console.log(c.name); //a,因为c.__proto__指向b,b.__proto__指向a,所以c可以访问到a的原型链上的属性 console.log(c.age); //undefined,因为在原型链上没找到age属性 console.log(c.__proto__.__proto__.__proto__.__proto__.__proto__); //null console.log(A.prototype === a.__proto__); //true,它们指向同一个原型对象 console.log(c.__proto__); //A {} console.log(b.__proto__); //A {} console.log(c === b); //false,c和b的原型链长度是不一样的,也就是说{}展开的内容是不一样的 三、总结1.是个对象就有原型 2.对象的原型是个对象或者null 3.被 [[Prototype]] 链接所引用的对象就是该对象的“原型” 4.构造函数.prototype指向一个原型对象,这个对象是构造函数的实例的原型 5.通过Object.getPrototypeOf或者__proto__逐级访问形成的链条叫做原型链 |
CopyRight 2018-2019 实验室设备网 版权所有 |