转载 最详尽的 JS 原型与原型链终极详解,没有「多是」。(一)

转自:https://www.jianshu.com/p/dee9f8b14771javascript

 

三篇文章都更新完毕,完整的剖析了 JS 原型与原型链,但愿经过这些教程能让你对 Javascript 这门语言理解的更透彻!php


一. 普通对象与函数对象

JavaScript 中,万物皆对象!但对象也是有区别的。分为普通对象和函数对象,Object 、Function 是 JS 自带的函数对象。下面举例说明html

var o1 = {}; var o2 =new Object(); var o3 = new f1(); function f1(){}; var f2 = function(){}; var f3 = new Function('str','console.log(str)'); console.log(typeof Object); //function console.log(typeof Function); //function console.log(typeof f1); //function console.log(typeof f2); //function console.log(typeof f3); //function console.log(typeof o1); //object console.log(typeof o2); //object console.log(typeof o3); //object 

在上面的例子中 o1 o2 o3 为普通对象,f1 f2 f3 为函数对象。怎么区分,其实很简单,凡是经过 new Function() 建立的对象都是函数对象,其余的都是普通对象。f1,f2,归根结底都是经过 new Function()的方式进行建立的。Function Object 也都是经过 New Function()建立的java

必定要分清楚普通对象和函数对象,下面咱们会经常用到它。ecmascript

二. 构造函数

咱们先复习一下构造函数的知识:函数

function Person(name, age, job) { this.name = name; this.age = age; this.job = job; this.sayName = function() { alert(this.name) } } var person1 = new Person('Zaxlct', 28, 'Software Engineer'); var person2 = new Person('Mick', 23, 'Doctor'); 

上面的例子中 person1 和 person2 都是 Person 的实例。这两个实例都有一个 constructor (构造函数)属性,该属性(是一个指针)指向 Person。 即:ui

console.log(person1.constructor == Person); //true console.log(person2.constructor == Person); //true 

咱们要记住两个概念(构造函数,实例):
person1 和 person2 都是 构造函数 Person 的实例
一个公式:
实例的构造函数属性(constructor)指向构造函数。this

三. 原型对象

在 JavaScript 中,每当定义一个对象(函数也是对象)时候,对象中都会包含一些预约义的属性。其中每一个函数对象都有一个prototype 属性,这个属性指向函数的原型对象。(先用无论什么是 __proto__ 第二节的课程会详细的剖析)spa

function Person() {} Person.prototype.name = 'Zaxlct'; Person.prototype.age = 28; Person.prototype.job = 'Software Engineer'; Person.prototype.sayName = function() { alert(this.name); } var person1 = new Person(); person1.sayName(); // 'Zaxlct' var person2 = new Person(); person2.sayName(); // 'Zaxlct' console.log(person1.sayName == person2.sayName); //true 

咱们获得了本文第一个「定律」:prototype

每一个对象都有 __proto__ 属性,但只有函数对象才有 prototype 属性

那什么是原型对象呢?
咱们把上面的例子改一改你就会明白了:

Person.prototype = {
   name: 'Zaxlct', age: 28, job: 'Software Engineer', sayName: function() { alert(this.name); } } 

原型对象,顾名思义,它就是一个普通对象(废话 = =!)。从如今开始你要紧紧记住原型对象就是 Person.prototype ,若是你仍是惧怕它,那就把它想一想成一个字母 A: var A = Person.prototype


在上面咱们给 A 添加了 四个属性:name、age、job、sayName。其实它还有一个默认的属性:constructor

在默认状况下,全部的原型对象都会自动得到一个 constructor(构造函数)属性,这个属性(是一个指针)指向 prototype 属性所在的函数(Person)

上面这句话有点拗口,咱们「翻译」一下:A 有一个默认的 constructor 属性,这个属性是一个指针,指向 Person。即:
Person.prototype.constructor == Person


在上面第二小节《构造函数》里,咱们知道实例的构造函数属性(constructor)指向构造函数person1.constructor == Person

这两个「公式」好像有点联系:

person1.constructor == Person
Person.prototype.constructor == Person

person1 为何有 constructor 属性?那是由于 person1 是 Person 的实例。
那 Person.prototype 为何有 constructor 属性??同理, Person.prototype (你把它想象成 A) 也是Person 的实例。
也就是在 Person 建立的时候,建立了一个它的实例对象并赋值给它的 prototype,基本过程以下:

var A = new Person(); Person.prototype = A; 

结论:原型对象(Person.prototype)是 构造函数(Person)的一个实例。


原型对象其实就是普通对象(但 Function.prototype 除外,它是函数对象,但它很特殊,他没有prototype属性(前面说道函数对象都有prototype属性))。看下面的例子:

function Person(){}; console.log(Person.prototype) //Person{} console.log(typeof Person.prototype) //Object console.log(typeof Function.prototype) // Function,这个特殊 console.log(typeof Object.prototype) // Object console.log(typeof Function.prototype.prototype) //undefined 

Function.prototype 为何是函数对象呢?

var A = new Function (); Function.prototype = A; 

上文提到凡是经过 new Function( ) 产生的对象都是函数对象。由于 A 是函数对象,因此Function.prototype 是函数对象。

那原型对象是用来作什么的呢?主要做用是用于继承。举个例子:

var Person = function(name){ this.name = name; // tip: 当函数执行时这个 this 指的是谁? }; Person.prototype.getName = function(){ return this.name; // tip: 当函数执行时这个 this 指的是谁? } var person1 = new person('Mick'); person1.getName(); //Mick 

从这个例子能够看出,经过给 Person.prototype 设置了一个函数对象的属性,那有 Person 的实例(person1)出来的普通对象就继承了这个属性。具体是怎么实现的继承,就要讲到下面的原型链了。

小问题,上面两个 this 都指向谁?

var person1 = new person('Mick'); person1.name = 'Mick'; // 此时 person1 已经有 name 这个属性了 person1.getName(); //Mick 

故两次 this 在函数执行时都指向 person1。


2017-10-27 更新:
下面的评论有喷子喷我:「null 没有 _proto_」,下面解释一下:
null 是一个独立的数据类型,为何typeof(null)的值是"object"?

  1. null不是一个空引用, 而是一个原始值, 参考ECMAScript5.1中文版** 4.3.11节; 它只是指望此处将引用一个对象, 注意是"指望", 参考 null - JavaScript**.
  2. typeof null结果是object, 这是个历史遗留bug, 参考 typeof - JavaScript**
  3. 在ECMA6中, 曾经有提案为历史平凡, 将type null的值纠正为null, 但最后提案被拒了. 理由是历史遗留代码太多, 不想得罪人, 不如继续将错就错当和事老, 参考 harmony:typeof_null [ES Wiki]
    做者:克荷林连接:https://www.zhihu.com/question/21691758/answer/987822

写个教程不容易,遇到喷子真心不爽。喷子无论那么多,哪怕文章是免费看的。他们只要抓住你一点点的小失误,就兴奋的噼里啪啦一顿喷。


别着急走,关于原型与原型链还有第二篇第三篇呢~
第二篇已更新,点击进入
第三篇已更新,点击进入


本文部分参照了 JS原型与原型链终极详解 (感谢原做者)(侵立删)
和《JavaScript 高级程序设计》第三版

但愿经过这一系列课程,能帮助你完全搞明白 JavaScript 的原型与原型链 :)

做者:Yi罐可乐 连接:https://www.jianshu.com/p/dee9f8b14771 來源:简书 简书著做权归做者全部,任何形式的转载都请联系做者得到受权并注明出处。
相关文章
相关标签/搜索