浅谈javascript中的prototype和__proto__的理解

2022-10-20,,,,

在工作中有时候会看到prototype__proto__这两个属性,对这两个属性我一直比较蒙圈,但是我通过查阅相关资料,决定做一下总结加深自己的理解,写得不对的地方还请各位大神指出。

  1. 跟__proto__属性相关的两个方法
  2. 判断属性是存在实例对象中,还是存在原型对象中的方法
  3. 获取或遍历对象中属性的几种方法

1、prototype

每个函数都有一个prototype属性,该属性是一个指针,指向一个对象。 而这个对象的用途是包含由特定类型的所有实例共享的属性和方法。使用这个对象的好处就是可以让所有实例对象共享它所拥有的属性和方法

2、 __proto__

每个实例对象都有一个__proto__属性,用于指向构造函数的原型对象。__proto__属性是在调用构造函数创建实例对象时产生的。

function person(name, age, job){ 
 this.name = name;
 this.age = age;
 this.job = job;
 this.sayname = function(){
 console.log(this.name);
 }; // 与声明函数在逻辑上是等价的
}
var person1=new person("nicholas",29,"software engineer");
console.log(person1);
console.log(person);
console.log(person1.prototype);//undefined
console.log(person1.__proto__);
console.log(person.prototype);
console.log(person1.__proto__===person.prototype);//true

输出结果如下:

总结:

1、调用构造函数创建的实例对象的prototype属性为"undefined",构造函数的prototype是一个对象。

2、__proto__属性是在调用构造函数创建实例对象时产生的。

3、调用构造函数创建的实例对象的__proto__属性指向构造函数的prototype。

4、在默认情况下,所有原型对象都会自动获得一个constructor(构造函数)属性,这个属性包含一个指向prototype属性所在函数的指针。

下图展示了使用person构造函数创建实例后各个对象之间的关系

上图展示了 person 构造函数、 person 的原型属性以及 person现有的两个实例之间的关系。

3、 跟__proto__属性相关的两个方法

isprototypeof():虽然在所有实现中都无法访问到__proto__,但可以通过 isprototypeof()方法来确定对象之间是否存在这种关系。

alert(person.prototype.isprototypeof(person1)); //true
 alert(person.prototype.isprototypeof(person2)); //true

object.getprototypeof():在所有支持的实现中,这个方法返回__proto__的值。例如:

alert(object.getprototypeof(person1) == person.prototype); //true
 alert(object.getprototypeof(person1).name); //"nicholas"

注意:虽然可以通过对象实例访问保存在原型中的值,但却不能通过对象实例重写原型中的值。如果我们在实例中添加了一个属性,而该属性与实例原型中的一个属性同名,那我们就在实例中创建该属性,该属性将会屏蔽原型中的那个属性。请看下面的例子:

function person(){
 }
 person.prototype.name = "nicholas";
 person.prototype.age = 29;
 person.prototype.job = "software engineer";
 person.prototype.sayname = function(){
 alert(this.name);
 };
 var person1 = new person();
 var person2 = new person();
 person1.name = "greg";
 alert(person1.name); //"greg"—— 来自实例
 alert(person2.name); //"nicholas"—— 来自原型

4、 判断属性是存在实例对象中,还是存在原型对象中,有以下方法

hasownproperty():可以检测一个属性是存在于实例中,还是存在于原型中。返回值为true表示该属性存在实例对象中,其他情况都为false。

in 操作符:无论该属性存在于实例中还是原型中。只要存在对象中,都会返回true。但是可以同时使用 hasownproperty()方法和 in 操作符,就可以确定该属性到底是存在于对象中,还是存在于原型中。

var person1 = new person();
 var person2 = new person();
 alert(person1.hasownproperty("name")); //false
 alert("name" in person1); //true
 person1.name = "greg";
 alert(person1.name); //"greg" —— 来自实例
 alert(person1.hasownproperty("name")); //true
 alert("name" in person1); //true
 alert(person2.name); //"nicholas" —— 来自原型
 alert(person2.hasownproperty("name")); //false
 alert("name" in person2); //true
 delete person1.name;
 alert(person1.name); //"nicholas" —— 来自原型
 alert(person1.hasownproperty("name")); //false
 alert("name" in person1); //true

5、 获取或遍历对象中属性的几种方法

for-in:通过for-in循环的返回的是能够被访问的、可枚举的属性,不管该属性是在实例中,还是存在原型中。

function person(name, age, job) {
		this.name = name;
		this.age = age;
		this.job = job;	
	}
	person.prototype={
		sayname:function(){
			return this.name;
		}
	}
	var p=new person("李明",30,"诗人");
	for(var prop in p){
		console.log(prop);//name、age、job、sayname
	}
 console.log(object.keys(p));//["name", "age", "job"]
 console.log(object.keys(person.prototype));//["sayname"]
 console.log(object.getownpropertynames(person.prototype))
 // ["constructor", "sayname"] 

object.keys():取得实例对象上所有可枚举的属性。 object.getownpropertynames(): 获取实例对象所有属性,无论它是否可枚举。

注意:使用对象字面量来重写整个原型对象时,本质上完全重写了默认的 prototype 对象,因此 constructor 属性也就变成了新对象的 constructor 属性(指向 object 构造函数),不再指向 person。但是可以通过在重写原型对象时指定constructor属性,使之还是指向原来的constructor。此时,尽管 instanceof 操作符还能返回正确的结果,但通过 constructor 已经无法确定对象的类型了。

object instanceof constructor:检测 constructor.prototype 是否存在于参数 object 的原型链上。

function person() {}
 var friend2 = new person();
 person.prototype = {
 	//constructor : person,
 	name: "nicholas",
 	age: 29,
 	job: "software engineer",
 	sayname: function() {
 		alert(this.name);
 	}
 };
 var friend = new person();
 console.log(friend2 instanceof object); //true
 console.log(friend2 instanceof person); //false,
 console.log(friend2.constructor == person); //true
 console.log(friend2.constructor == object); //false
 
 console.log(friend instanceof object); //true
 console.log(friend instanceof person); //true
 console.log(friend.constructor == person); //false
 console.log(friend.constructor == object); //true

由于原型的动态性,调用构造函数时会为实例添加一个指向最初原型的prototype指针,而把原型修改为另外一个对象就等于切断了构造函数与最初原型之间的联系。看下面的例子

function person(){
 }
 var friend = new person();
 person.prototype = {
 constructor: person,
 name : "nicholas",
 age : 29,
 job : "software engineer",
 sayname : function () {
  alert(this.name);
 }
 };
 var friend2=new person();
 friend.sayname(); //uncaught typeerror: friend.sayname is not a function 
 friend2.sayname();//nicholas
 console.log(friend instanceof person);//false
 console.log(friend instanceof object);//true
 console.log(friend2 instanceof person);//true

结果分析:这是因为friend1的prototype指向的是没重写person.prototype之前的person.prototype,也就是构造函数最初的原型对象。而friend2的prototype指向的是重写person.prototype后的person.prototype。如下图所示

 

6、 原型链

基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。最直观的表现就是让原型对象等于另一个类型的实例。

function supertype(){
 this.property = true;
 }
 supertype.prototype.getsupervalue = function(){
 return this.property;
 };
 function subtype(){
 this.subproperty = false;
 }
 //继承了 supertype
 subtype.prototype = new supertype();
 subtype.prototype.getsubvalue = function (){
 return this.subproperty;
 };
 var instance = new subtype();
 alert(instance.getsupervalue()); //true

subtype.prototype = new supertype();这句代码使得原来存在于 supertype 的实例中的所有属性和方法,现在也存在于 subtype.prototype 中。使得instance的constructor指向了supertype。

console.log(instance.constructor===supertype);//true

总结: 访问一个实例属性时,首先会在实例中搜索该属性。如果没有找到该属性,则会继续搜索实例的原型。在通过原型链实现继承的情况下,搜索过程就得以沿着原型链继续向上。在找不到属性或方法的情况下,搜索过程总是要一环一环地前行到原型链末端才会停下来。

就拿上面的例子来说,调用 instance.getsupervalue()会经历4个搜索步骤:

  1. 搜索instance实例;
  2. 搜索 subtype.prototype;
  3. 搜索supertype的实例;
  4. 搜索 supertype.prototype,最后一步才会找到该方法。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

《浅谈javascript中的prototype和__proto__的理解.doc》

下载本文的Word格式文档,以方便收藏与打印。