五种继承的方式

1. 原型链继承

核心:让子类的原型指向父类的实例。

js 复制代码
function Parent(name) {
  this.name = name;
  this.colors = ["red", "blue"];
}
Parent.prototype.sayHi = function() {
  console.log("Hi from Parent");
};

function Child() {}
Child.prototype = new Parent("Tom"); // 关键

const c1 = new Child();
c1.colors.push("green");

const c2 = new Child();
console.log(c2.colors); // ["red","blue","green"] ❌ 被共享了

console.log(c1.name); // "Tom"
console.log(c2.name); // "Tom"

问题出在哪?

关键是 调用 new Child() ,并没有再执行 Parent 构造函数。

  • Parent("Tom") 只在 设置原型链的时候被调用了一次。
  • 之后无论你创建多少个 Child 实例,都会复用这个"已经带着 name=Tom 的对象"。
  • 所以 Child 实例的 name 永远都是 "Tom",你没法在 new Child("Jerry") 的时候传递参数给 Parent

举例说明

vbscript 复制代码
const c1 = new Child("Jerry");
console.log(c1.name); // ❌ 还是 "Tom",根本没用 "Jerry"

为什么?

因为 Child 构造函数里面压根没调用 Parent,也就没法把 "Jerry" 传过去。

缺点

  • 父类实例属性(引用类型)会被所有子类实例共享。
  • 不能向父类构造函数传参。

2. 借用构造函数继承(经典继承) ----解决原型继承 实例属性共享问题

核心:在子类构造函数里调用父类构造函数。

js 复制代码
function Parent(name) {
  this.name = name;
  this.colors = ["red", "blue"];
}
Parent.prototype.sayName = function(){
    console.log(this.name)
}
function Child(name) {
  Parent.call(this, name); // 关键
}

const c1 = new Child("Tom");
c1.colors.push("green");

const c2 = new Child("Jerry");
console.log(c2.colors); // ["red","blue"] ✅ 独立
c2.sayName() // ❌ c2.sayName is not a function

优点

  • 解决了原型链继承共享引用属性的问题。
  • 可以给父类构造函数传参。

缺点

  • 方法都在构造函数里定义,没法复用,浪费内存。// colors在每个子类实例上都有一份
  • 不能继承父类原型上的方法

本质区别总结

  • 原型链继承 :共享的是 父类实例对象 上的属性 → 所有子类实例共用同一个引用,new Parent() 只执行了一次(在设置 Child.prototype 时),所以只分配了一份存储空间。
  • 构造函数继承 :每次实例化时重新执行父类构造函数 → 每个子类实例独立拥有自己的属性,Parent.call(this) 在每个子类实例里都会执行一次,所以每次 new Child() 都会分配一份新的存储空间。

3. 组合继承(原型链 + 构造函数) ✅ 常用

核心:结合两者优点。

js 复制代码
function Parent(name) {
  this.name = name;
  this.colors = ["red", "blue"];
}
Parent.prototype.sayHi = function() {
  console.log("Hi, I am " + this.name);
};

function Child(name, age) {
  Parent.call(this, name); // 借用构造函数
  this.age = age;
}
Child.prototype = new Parent(); // 原型链
Child.prototype.constructor = Child;

const c1 = new Child("Tom", 18);
c1.sayHi(); // Hi, I am Tom

优点

  • 既能复用方法,又能独立拥有属性。

缺点

  • 父类构造函数会被调用两次(一次在 Parent.call,一次在 new Parent())。

4. 寄生组合继承(最推荐的 ES5 写法)

js 复制代码
function Animal(name) {
  this.name = name;
}
Animal.prototype.sayHi = function() {
  console.log(`I am ${this.name}`);
};

function Dog(name) {
  Animal.call(this, name);
}
// 用 Object.create 避免多次调用 Animal
Dog.prototype = Object.create(Animal.prototype);
Dog.prototype.constructor = Dog;

const d = new Dog("Lucky");
d.sayHi(); // I am Lucky

🔹 特点:

  • 完美解决了组合继承的缺点。
  • ES5 最佳实践

5. ES6 class 继承(语法糖)

js 复制代码
class Animal {
  constructor(name) {
    this.name = name;
  }
  sayHi() {
    console.log("I am " + this.name);
  }
}

class Dog extends Animal {
  constructor(name, color) {
    super(name); // 调用父类构造函数,相当于 Animal.call(this, name)
    this.color = color;
  }
  bark() {
    console.log("Woof!");
  }
}

const d = new Dog("Lucky", "white");
d.sayHi(); // I am Lucky
d.bark();  // Woof!

super(name) 的作用:

  1. 调用父类构造函数 Animal(name)
  2. 把返回的结果绑定到当前子类实例的 this。 🔹 特点:
  • 语法更简洁直观。
  • 内部还是基于原型链。
相关推荐
袋鼠云数栈UED团队1 天前
基于 Lexical 实现变量输入编辑器
前端·javascript·架构
cipher1 天前
ERC-4626 通胀攻击:DeFi 金库的"捐款陷阱"
前端·后端·安全
UrbanJazzerati1 天前
非常友好的Vue 3 生命周期详解
前端·面试
AAA阿giao1 天前
从零构建一个现代登录页:深入解析 Tailwind CSS + Vite + Lucide React 的完整技术栈
前端·css·react.js
兆子龙1 天前
像 React Hook 一样「自动触发」:用 Git Hook 拦住忘删的测试代码与其它翻车现场
前端·架构
兆子龙1 天前
用 Auto.js 实现挂机脚本:从找图点击到循环自动化
前端·架构
SuperEugene1 天前
表单最佳实践:从 v-model 到自定义表单组件(含校验)
前端·javascript·vue.js
昨晚我输给了一辆AE861 天前
为什么现在不推荐使用 React.FC 了?
前端·react.js·typescript
不会敲代码11 天前
深入浅出 React 闭包陷阱:从现象到原理
前端·react.js
不会敲代码11 天前
React性能优化:深入理解useMemo和useCallback
前端·javascript·react.js