JavaScript原型和原型链是什么意思
小编给大家分享一下JavaScript原型和原型链是什么意思,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!
一、原型、原型链相等关系理解
首先我们要清楚明白两个概念:
js
分为函数对象和普通对象,每个对象都有__proto__
属性,但是只有函数对象才有prototype
属性Object
、Function
都是js内置的函数, 类似的还有我们常用到的Array
、RegExp
、Date
、Boolean
、Number
、String
这两个概念大家跟我一起读三遍并记住,后面会用到
那么__proto__
和prototype
到底是什么,两个概念理解它们
属性
__proto__
是一个对象,它有两个属性,constructor
和__proto__
;原型对象
prototype
有一个默认的constructor
属性,用于记录实例是由哪个构造函数创建;
这两个概念大家跟我一起读三遍并记住,后面会用到
有以下构造函数
Person
,他的原型上有所属国属性motherland='China'
function Person(name, age){ this.name = name; this.age = age; } Person.prototype.motherland = 'China'
通过
new Person()
创建的person01
实例
let person01 = new Person('小明', 18);
js之父在设计js原型、原型链的时候遵从以下两个准则
Person.prototype.constructor == Person // **准则1:原型对象(即Person.prototype)的constructor指向构造函数本身** person01.__proto__ == Person.prototype // **准则2:实例(即person01)的__proto__和原型对象指向同一个地方**
这两个准则大家跟我一起读三遍并记住,后面会用到
记住以上四个概念两个准则,任何原型链相等判断都能判断正确;
大家可以对照上图,看看自己概念准则是否弄清楚了,一定要对照上图哦
// 从上方 function Foo() 开始分析这一张经典之图function Foo()let f1 = new Foo();let f2 = new Foo();f1.__proto__ = Foo.prototype; // 准则2f2.__proto__ = Foo.prototype; // 准则2Foo.prototype.__proto__ = Object.prototype; // 准则2 (Foo.prototype本质也是普通对象,可适用准则2)Object.prototype.__proto__ = null; // 原型链到此停止Foo.prototype.constructor = Foo; // 准则1Foo.__proto__ = Function.prototype; // 准则2Function.prototype.__proto__ = Object.prototype; // 准则2 (Function.prototype本质也是普通对象,可适用准则2)Object.prototype.__proto__ = null; // 原型链到此停止// **此处注意Foo 和 Function的区别, Foo是 Function的实例**// 从中间 Function Object()开始分析这一张经典之图Function Object()let o1 = new Object();let o2 = new Object();o1.__proto__ = Object.prototype; // 准则2o2.__proto__ = Object.prototype; // 准则2Object.prototype.__proto__ = null; // 原型链到此停止Object.prototype.constructor = Object; // 准则1Object.__proto__ = Function.prototype // 准则2 (Object本质也是函数);// 此处有点绕,Object本质是函数,Function本质是对象Function.prototype.__proto__ = Object.prototype; // 准则2 (Function.prototype本质也是普通对象,可适用准则2)Object.prototype.__proto__ = null; // 原型链到此停止// 从下方 Function Function()开始分析这一张经典之图Function Function()Function.__proto__ = Function.prototype // 准则2Function.prototype.constructor = Function; // 准则1
由此可以得出结论: 除了Object
的原型对象(Object.prototype
)的__proto__
指向null
,其他内置函数对象的原型对象(例如:Array.prototype)和自定义构造函数的 __proto__
都指向Object.prototype
, 因为原型对象本身是普通对象。 即:
Object.prototype.__proto__ = null;Array.prototype.__proto__ = Object.prototype;Foo.prototype.__proto__ = Object.prototype;
二:原型、原型链的意思何在
理解了这些相等关系之后,我们思考,原型、原型链的意思何在?原型对象的作用,是用来存放实例中共有的那部份属性、方法,可以大大减少内存消耗。
用我们文章开始的Person
构造函数和person01实例举例说:
console.log(person01)
打印person01
, 他有自己属性 name = '小明',age = 18; 同时通过原型链关系,他有属性motherland = 'China';
我们再创建person2实例
let person02 = new Person('小花', 20);console.log(person02)
打印person02
, 他有自己属性 name = '小花',age = 20; 同时通过原型链关系,他有属性motherland = 'China'
; 看出来了没有,原型对象存放了person01
、person02
共有的属性所属国motherland = 'China'
. 我们不用在每个实例上添加motherland 属性,而是将这一属性存在他们的构造函数原型对象上,对于人类Person这样的构造函数。相同的属性、方法还有很多很多,比如我们是黑头发,我们都有吃,睡这样一个方法,当相同的属性、方法越多,原型、原型链的意义越大。 那我们可以这样操作
Person.prototype.hairColor = 'black';Person.prototype.eat = function(){ console.log('We usually eat three meals a day.')}console.log(person01)console.log(person02)
此时我们再打印person01
、person02
,我们惊喜的发现,他们有了属性hairColor
和eat
方法;实例们动态的获得了Person
构造函数之后添加的属性、方法,这是就是原型、原型链的意义所在!可以动态获取,可以节省内存。
另外我们还要注意:如果person01将头发染成了黄色,那么
hairColor
会是什么呢?
person01,hairColor = 'yellow';console.log(person01)console.log(person02)
可以看到,person01
的hairColor = 'yellow'
, 而person02
的hairColor = 'black'
; 实例对象重写原型上继承的属相、方法,相当于"属性覆盖、属性屏蔽",这一操作不会改变原型上的属性、方法,自然也不会改变由统一构造函数创建的其他实例,只有修改原型对象上的属性、方法,才能改变其他实例通过原型链获得的属性、方法。
以上是"JavaScript原型和原型链是什么意思"这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注行业资讯频道!