构造函数详解(类比Vue构造函数)

2023-03-03  本文已影响0人  扶得一人醉如苏沐晨

一、典型的面向对象编程语言(比如C++和Java)

存在类(class)这个概念。所谓类就是对象的模板,对象就是类的实例。
但是,在JavaScript语言的对象体系,不是基于“类”的,而是基于构造函数(constructor)和原型链(prototype)

二、构造函数和对象的关系?

对象是单个实物的抽象,通常需要一个模板生成
js语言中使用构造函数(constructor)作为对象的模板,
构造函数,就是提供一个生成对象的模板,并描述对象的基本结构的函数。一个构造函数,可以生成多个对象,每个对象都有相同的结构。

三、构造函数作用和特点

3.1、作用

构造新对象,设置对象的属性和方法

3.2、特点

构造函数一般首字母会大写,为了和普通函数区分
一个构造函数可以通过new创建多个实例对象
创建构造函数时,里面的属性和方法前必须加this,this就表示当前运行时的对象

四、ECMAScript提供了多个内置构造函数,

如 Object、Array、String、Boolean、Number、Date…等等。
var obj = new Object();
var arr = new Array();
ECMAScript也允许自定义构造函数

五、构造函数的创建和使用

    function Person(name, height) {
          this.name = name;
          this.height = height;
          this.bark = function(fs){
              return fs
          }
     }
      var boy = new Person('Keith', 180);
      console.log(boy);  //Person {name: 'Keith', height: 180, bark: ƒ}
      console.log(boy.constructor);  //f Person(){}  //整个构造函数原型
      console.log(boy.bark(8));  //8
      console.log(boy.name); //'Keith'
      console.log(boy.height); //180

六、构造函数的return

构造函数不需要return 就可以返回结果

6.1、return一个基本数据类型

结果不变,依旧返回一个对象

  function Dog(){
      this.name = "贝贝";
      this.bark = function(){
          console.log("汪汪汪");
      }
      return 0;
  }
  var d1 = new Dog();
  console.log(d1);//Dog {name: '贝贝', bark: ƒ}

return一个复杂数据类型

返回一个复杂数据类型例:

 function Dog(){
      this.name = "贝贝";
      this.bark = function(){
          console.log("汪汪汪");
      }
      return [];
  }
  var d1 = new Dog();
  console.log(d1); // []

七、构造函数的原理(new之后发生了什么)

构造函数之所以能构造出对象,其实JS帮助我们做了很多骚操作。你以为new之后直接执行函数体代码,其实并不是,事实比我们看到了多了四步
①创建一个空对象newObj = { }
newObj.__proto__= Fn.prototype
③函数的this指向newObj
④执行代码
⑤默认返回newObj

1、自从用new调用函数后,JS引擎就会在内存中创建一个空对象{}
const newObj = {};
2、新对象的__proto__属性指向构造函数的原型对象
(通俗理解就是新对象隐式原型__proto__链接到构造函数显式原型prototype上。)

  newObj.__proto__ = FunctionName.prototype

3、构造函数内部的this会指向这个新对象(即将构造函数的作用域指向新对象)
this = newObj
4、从上到下执行函数体(只有这步是我们能直观看到代码的)
5、返回创造出来的对象(如果构造函数没有返回对象,则默认返回this。在函数体内部的this指向新创建的内存空间,默认返回 this 就相当于默认返回了该内存空间

function Person(name, age) {
    this.name = name;
    this.age = age;
    this.eating = function() {
        console.log(this.name + ' is eating');
    }
}

const p1 = new Person('zs', 12);

//----------------------------------------------------------------------------
/*实际JS引擎帮助我们实现的操作*/
const newObj = {};
newObj.__proto__ = Person.prototype;
this = newObj;

this.name = name;
this.age = age;
this.eating = function() {
  console.log(this.name + ' is eating');
}

return newObj;

八、在构造函数原型上绑定方法(重点!!!!!)

function Person(name, age) {
    this.name = name;
    this.age = age;
}
const p1 = new Person('zs', 12);
// 在函数原型上添加方法
Person.prototype.eating = function() {
    console.log(this.name + ' is eating'); // zs is eating
    return 5
}
console.log(p1.eating());  //5

将方法转移到构造函数原型上来定义后,对于实例对象p1,依然可以调用eating方法。调用p1的eating方法时,如果p1对象没有该方法,会去p1对象的原型对象p1.__proto_找,因为在构造p1时,绑定的原型:p1.__proto__ = Person.prototype,所以可以找到p1.__proto__.eating

九、类比Vue构造函数

在vue中我们拥有Vue()和VueComponents()两个构造函数

9.1、Vue()构造函数

①const vm = new Vue()
new关键词帮我们做了至关重要的一步
vm.__protp__ = Vue.prototype

这个vm实例对象就是我们vue中最大的那个组件所有的子组件都是通过vm上面的VueComponents()构造函数new出来的

9.2、VueComponents()构造函数

在vm中,有一个VueComponents()

②const vc = new VueComponents()

vc.__protp__ = VueComponents.prototype

这里的vc就是我们最大的vm实例对象中的VueComponents()构造出来的

9.3、通过原型链一层一层往上面找一定可以找到我们Vue.prototype的方法和函数,所以我们在vue中每个组件都可以访问我们Vue.prototype上面的属性和方法

上一篇下一篇

猜你喜欢

热点阅读