开始

  1. Array.prototype.add = function(a){
  2. this.push(a)
  3. }
  4. var ss = new Array(5).fill(0); //[0,0,0,0,0]
  5. ss.add('1') //[0,0,0,0,0,"1"]

原型链实际上是JavaScript中的实现继承的机制,在搞懂原型链之前首先要搞懂几个概念:this,普通对象和函数对象,构造函数,new

this

  • this只能在存在与函数中
  • this其实是当前函数所在上下文环境,再简单一点也可以理解为this返回一个当前函数所在的对象,也就是说想要知道this是什么我们只需要关注是谁调用了this所在的函数就可以了

如下边的代码zhao.sayName()是zhao调用的sayName函数所以sayName中的this自然指的就是zhao这个对象,而下方 var liName = zhao.sayName语句是将sayName这个函数赋值给liName,调用liName就相当于在最顶层也就是window下直接调用sayName,this的指向自然就是window这个最顶层对象

  1. var name = "Li"
  2. var zhao = {
  3. name: "Zhao",
  4. sayName: function () {
  5. console.log(this.name);
  6. }
  7. }
  8. zhao.sayName() // Zhao
  9. var liName = zhao.sayName;
  10. liName() // Li


普通对象与函数对象

JavaScript中一切都可以看作对象,但是实际上对象也是有区别的,对象分为普通对象函数对象

  1. // 普通对象
  2. var o1 = {}
  3. var o2 = new Object()
  4. var o3 = new f1()
  5. // 函数对象
  6. function f1(){}
  7. var f2 = function(){}
  8. var f3 = new Function()
  9. console.log(typeof f1); //function
  10. console.log(f1.prototype); //true
  11. console.log(typeof f2); //function
  12. console.log(f2.prototype); //true
  13. console.log(typeof f3); //function
  14. console.log(f3.prototype); //true
  15. console.log(typeof o1); //object
  16. console.log(o1.prototype); //undefined
  17. console.log(typeof o2); //object
  18. console.log(o2.prototype); //undefined
  19. console.log(typeof o3); //object
  20. console.log(o3.prototype); //undefined

凡是通过function构建的对象都是函数对象,并且只有函数对象才有prototype属性,普通对象没有

prototype 原型

prototype又是什么呢?
当我们创建函数的时候,编译器会自动为该函数创建一个prototype属性,这和属性指向一个包含constructor属性的对象,而这个属性又默认指回原函数,读起来有点绕对吧,大概是这样的

  1. function Person() {
  2. // prototype = {
  3. // constructor: Person,
  4. // }
  5. }

每个函数对象都有一个prototype(原型)属性,在我看来prototype属性的意义:

  1. 创建对象的模板
  2. 公开的共享空间

这两点等学习了下边new命令你就会明白了

constructor 构造函数

函数对象的一种用法就是构造函数,通过构造函数可以构建一个函数对象的实例(普通对象)

  1. function Person(name, age ){
  2. this.name = name;
  3. this.age = age;
  4. this.sayHello = function(){
  5. console.log(`Hello! my name is ${this.name}`);
  6. };
  7. }
  8. var person1 = new Person("kidder", 28);
  9. person1.sayHello(); // Hello! my name is kidder
  10. console.log(person1.constructor); //[Function:Person]

按照惯例,构造函数的命名以大写字母开头,非构造函数以小写字母开头,通过构造函数构造的普通对象都会有一个constructor(构造函数)属性,该属性指向构造该对象的构造函数

new命令

new命令的工作机制

  1. 创建一个空对象作为要返回对象的实例
  2. 将这个空对象的原型( proto )指向构造函数的prototype属性
  3. 将这个空对象赋值给构造函数内部的this
  4. 执行构造函数内部的代码

原型链

下面我们来看看构造函数构建一个普通对象的时候发生了什么

  1. var Person = function (name) {
  2. this.name = name;
  3. this.age = 18;
  4. };
  5. Person.prototype.sayHello = function(){
  6. console.log(`Hello! my name is ${this.name}`);
  7. };
  8. var li = new Person("Li");
  9. console.log(li.name); // Li
  10. console.log(li.age); // 18
  11. li.sayHello(); // Hello! my name is Li
  1. 创建一个空对象作为要返回对象的实例

    1. {}
  2. 将这个空对象的原型( proto )指向构造函数的prototype属性

    1. {
    2. __proto__:Person.prototype;
    3. }
  3. 将这个空对象赋值给构造函数内部的this

    1. this = {
    2. __proto__:Person.prototype;
    3. }
  4. 执行构造函数内部的代码

    1. this = {
    2. __proto__:Person.prototype;
    3. name: "Li";
    4. age: 18;
    5. }

这就是原型链,当给定的属性在当前对象中找不到的情况下,会沿着proto这个属性一直向对象的上游去寻找,直到proto这个属性指向null为止,如果找到指定属性,查找就会被截断,停止
原型链 - 图1
上面这张图是整个JavaScript的原型链体系,为了让这张图更直观所以我将构造函数的prototype属性单独提了出来,恩,其实画在构造函数内部也可,但同时因为对象是引用类型,所以这样画也没毛病吧

proto 和 prototype

prototype:只有函数对象才具有的属性,它用来存放的是构造函数希望构建的实例具有的共享的属性和方法,主要用于构造函数的实例化

proto : 所有对象都具有的属性,它指向的是当前对象在原型链上的上级对象,主要作用是让编译器在由proto这个属性构成的原型链上查找特定的属性和方法

补充

prototype的共享属性

  1. var Person = function (name) {
  2. this.name = name;
  3. this.age = 18;
  4. };
  5. Person.prototype.sayHello = function(){
  6. console.log(`Hello! my name is ${this.name}`);
  7. };
  8. var li = new Person("Li");
  9. var Person1 = function () {
  10. };
  11. Person.prototype.name = "Li"
  12. Person.prototype.age = 18
  13. Person.prototype.sayHello = function(){
  14. console.log(`Hello! my name is ${this.name}`);
  15. };
  16. var Li = new Person1();

关于Person和Person1两种构造函数的写法有什么不同呢?
原型链 - 图2
一般来说写在prototype原型对象中的属性和方法都是公用的,也就是说写在构造函数中的属性在构建普通对象的时候,都会在新对象中重新定义,也就是从内存的角度来说又会多占用一些内存空间,所以我们将构造函数的所有属性和方法都写在prototype原型中不好吗?

但是原型函数也是有缺点的:

  1. 不够灵活

    1. var Person = function () {};
    2. Person.prototype.name = "Li"
    3. Person.prototype.age = 18
    4. Person.prototype.sayHello = function(){
    5. console.log(`Hello! my name is ${this.name}`);
    6. };
    7. var li = new Person();
    8. var zhao = new Person();

    这种方式构造的所有对象都是一个模板,虽然我们也可以在当前对象下进行修改,但这样一点也不优雅,不规整,而且从某种意义上来说也是对内存的浪费

  2. 对于引用类型的修改会被全部共享

    1. var Person = function () {};
    2. Person.prototype.name = "Li"
    3. Person.prototype.age = 18
    4. Person.prototype.friends = ["ZhangSan", "LiSi"]
    5. Person.prototype.sayHello = function(){
    6. console.log(`Hello! my name is ${this.name}`);
    7. };
    8. var li = new Person();
    9. var zhao = new Person();
    10. li.friends.push("WangWu");
    11. console.log(zhao.friends); // [ 'ZhangSan', 'LiSi', 'WangWu' ]

    在JavaScript中,基本类型的修改可以明确的通过创建或修改在当前对象下的属性对原型链进行截断,但是像数组,对象这种引用类型的值虽然也可以通过在当前对象中创建该属性来对原型链进行截断,但是一不注意就可能会出现上面这种情况直接对原型进行了修改

构造函数与原型相结合

所以,用构造函数来定义实例属性,用原型定义方法和共享的属性,这样写就比较优雅了

  1. function Person(name, age){
  2. this.name = name;
  3. this.age = age;
  4. this.friends = ["ZhangSan", "LiSi"];
  5. }
  6. Person.prototype.sayHello = function(){
  7. console.log(`Hello! my name is ${this.name},${this.age}岁了`);
  8. };
  9. var li = new Person("li", 18);
  10. var zhao = new Person("zhao", 16);
  11. li.sayHello(); // Hello! my name is li, 18岁了
  12. zhao.sayHello(); // Hello! my name is zhao,16岁了
  13. li.friends.push("WangWu");
  14. console.log(zhao.friends); // [ 'ZhangSan', 'LiSi' ]

创建对象的几种方式

  1. 构造函数方式

法一用构造函数构造一个新对象

  1. var A = function () { };
  2. var a = new A();
  3. console.log(a.constructor); // [Function:A]
  4. console.log(a.__proto__ === A.prototype); //true

原型链 - 图3

  1. 字面量方式

法二的本质来说和法一是一样的,就是隐式调用原生构造函数Object来构造新对象

  1. var a = {};
  2. // var a = new Object();
  3. console.log(a.constructor); // [Function:Object]
  4. console.log(a.__proto__ === Object.prototype); //true

原型链 - 图4

  1. create方式

法三Object.create是以一个普通对象为模板创建一个新对象

  1. var a1 = {a:1}
  2. var a2 = Object.create(a1);
  3. console.log(a2.constructor); // [Function:Object]
  4. console.log(a2.__proto__ === a1);// true
  5. console.log(a2.__proto__ === a1.prototype); //false

原型链 - 图5
所以除了Object.create创建对象的方式,可以说: proto === constructor.prototype;

constructor

前面我们说道prototype的时候进行原型属性的赋值的时候,采用的是逐项赋值,那么当我直接将对象赋值给prototype属性的时候会发生什么呢?

  1. function Person() { }
  2. Person.prototype = {
  3. name : "Li",
  4. age : 18,
  5. sayHello : function () {
  6. console.log(`Hello! my name is ${this.name},${this.age}岁了`);
  7. }
  8. };
  9. var li = new Person();
  10. console.log(li instanceof Object); // true
  11. console.log(li instanceof Person); // true
  12. console.log(li.constructor === Person); // false
  13. console.log(li.constructor === Object); // true
  14. console.log(Person.prototype.constructor); // Object

这时候我们就发现我们构建的li对象的constructor不再指向它的构造函数Person,而是指向了Object,并且Person原型Person.prototype的constructor指向也指向了Object,这是什么原因呢?
其实,根源出现在Person.prototype上,上边我们提到过,其实我们在写构造函数的时候实际上是这样的

  1. function Person() {
  2. // prototype = {
  3. // constructor : Person
  4. // }
  5. }

当我们构建Person构造函数的时候,编译器会自动生成一个带有指向Person的constructor属性的对象,并把这个对象赋值给Person.prototype,我们又知道js中对象是引用类型,当我们使用Person.prototype.name=…的时候实际上是对这个对象的修改,而使用Person.prototype={…}实际上是将这个属性原本的指针指向了另一个新创建的对象而不是原来编译器自动创建的那个:
原型链 - 图6
而li的constructor属性自然是继承自Person.prototype,所以constructor自然也就跟着改变了,如果在编程的过程中constructor这个属性很重要的话可以通过下面的方式

  1. function Person() { }
  2. Person.prototype = {
  3. constructorPerson
  4. name : "Li",
  5. age : 18,
  6. sayHello : function () {
  7. console.log(`Hello! my name is ${this.name},${this.age}岁了`);
  8. }
  9. };
  10. var li = new Person();
  11. console.log(li instanceof Object); // true
  12. console.log(li instanceof Person); // true
  13. console.log(li.constructor === Person); // true
  14. console.log(li.constructor === Object); // false
  15. console.log(Person.prototype.constructor); // Person

参考

  1. js回顾:原型链