JavaScript面试:类继承和原型继承区别?

2021-02-23  本文已影响0人  魂斗驴

对象经常在JavaScript中使用,而了解如何有效地使用它们将为您的工作效率带来巨大的胜利。实际上,糟糕的OO设计可能会导致项目失败,在最坏的情况下还会导致公司失败

与大多数其他语言不同,JavaScript的对象系统基于原型,而不是类。不幸的是,大多数JavaScript开发人员都不了解JavaScript的对象系统或如何最好地利用它。其他人的确理解它,但是希望它的行为更像基于类的系统。结果是JavaScript的对象系统具有令人困惑的拆分个性,这意味着JavaScript开发人员需要对原型和类有所了解。

类和原型继承之间有什么区别?

类的继承: 类就像对要创建的对象的描述。类从类继承并创建子类关系

实例通常通过带有new关键字的构造函数来实例化。类继承可以使用也可以不使用ES6中的class关键字。您可能从Java之类的语言中了解到的类在技术上并不存在于JavaScript中。而是使用构造函数。该ES6 class关键字转换成构造函数:

class Foo {}
typeof Foo // `function`

在JavaScript中,类继承是在原型继承的基础上实现的,但这并不意味着它执行相同的操作:
JavaScript的类继承使用原型链将子项“ Constructor.prototype”连接到父项“ Constructor.prototype”进行代理。通常,也调用super()构造函数。这些步骤形成了单祖先的父/子层次结构,并创建了OO设计中可用的最紧密的耦合。

“类从类继承并创建子类关系”

原型继承: 原型是工作对象实例。 对象直接从其他对象继承。

实例可以由许多不同的源对象组成,从而可以轻松地进行选择性继承并实现平坦的[[Prototype]]代理层次结构。换句话说,分类法并不是原型OO的自动副作用:一个关键的区别。

实例通常通过工厂函数,对象,或 Object.create()实例化。

“原型是一个工作对象实例。对象直接从其他对象继承。”

为什么如此重要?

继承从根本上讲是一种代码重用机制:一种不同类型的对象共享代码的方式。共享代码的方式很重要,因为如果您弄错了代码,则会造成很多问题,特别是:

类继承将父/子对象分类法作为副作用。

这些分类法实际上不可能适用于所有新的用例,并且基类的广泛使用会导致脆弱的基类问题,这会使它们在错误时难以修复。 实际上,类继承会在OO设计中引起许多众所周知的问题:

所有继承都是不好的吗?

这是OO设计中的常识,因为类继承存在许多缺陷并引起许多问题。人们通常在谈论类继承时会忽略类一词,这听起来似乎所有继承都是不好的,但事实并非如此。

实际上,有几种不同的继承,其中大多数对于从多个组件对象组成复合对象非常有用。

三种不同的原型继承

在深入探讨其他类型的继承之前,让我们仔细研究一下类继承的含义:

// Class Inheritance Example
// NOT RECOMMENDED. Use object composition, instead.

// https://gist.github.com/ericelliott/b668ce0ad1ab540df915
// http://codepen.io/ericelliott/pen/pgdPOb?editors=001

class GuitarAmp {
  constructor ({ cabinet = `spruce`, distortion = `1`, volume = `0` } = {}) {
    Object.assign(this, {
      cabinet, distortion, volume
    });
  }
}

class BassAmp extends GuitarAmp {
  constructor (options = {}) {
    super(options);
    this.lowCut = options.lowCut;
  }
}

class ChannelStrip extends BassAmp {
  constructor (options = {}) {
    super(options);
    this.inputLevel = options.inputLevel;
  }
}

test(`Class Inheritance`, nest => {
  nest.test(`BassAmp`, assert => {
    const msg = `instance should inherit props
    from GuitarAmp and BassAmp`;

    const myAmp = new BassAmp();
    const actual = Object.keys(myAmp);
    const expected = [`cabinet`, `distortion`, `volume`, `lowCut`];

    assert.deepEqual(actual, expected, msg);
    assert.end();
  });

  nest.test(`ChannelStrip`, assert => {
    const msg = `instance should inherit from GuitarAmp, BassAmp, and ChannelStrip`;
    const myStrip = new ChannelStrip();
    const actual = Object.keys(myStrip);
    const expected = [`cabinet`, `distortion`, `volume`, `lowCut`, `inputLevel`];

    assert.deepEqual(actual, expected, msg);
    assert.end();
  });
});

在CodePen 中运行

“ BassAmp”继承自“ GuitarAmp”,而“ ChannelStrip”继承自“ BassAmp”和“ GuitarAmp”。这是OO设计出错的一个例子。通道条实际上不是吉他放大器的一种,并且实际上根本不需要cabinet。更好的选择是创建一个新的基类,ampschannel都可以继承该基类,但即使这样也有局限性。

最终,新的共享基类策略也崩溃了。

有更好的方法。您可以使用对象组合继承您真正需要的东西:

// Composition Example

// http://codepen.io/ericelliott/pen/XXzadQ?editors=001
// https://gist.github.com/ericelliott/fed0fd7a0d3388b06402

const distortion = { distortion: 1 };
const volume = { volume: 1 };
const cabinet = { cabinet: `maple` };
const lowCut = { lowCut: 1 };
const inputLevel = { inputLevel: 1 };

const GuitarAmp = (options) => {
  return Object.assign({}, distortion, volume, cabinet, options);
};

const BassAmp = (options) => {
  return Object.assign({}, lowCut, volume, cabinet, options);
};

const ChannelStrip = (options) => {
  return Object.assign({}, inputLevel, lowCut, volume, options);
};


test(`GuitarAmp`, assert => {
  const msg = `should have distortion, volume, and cabinet`;
  const level = 2;
  const cabinet = `vintage`;

  const actual = GuitarAmp({
    distortion: level,
    volume: level,
    cabinet
  });
  const expected = {
    distortion: level,
    volume: level,
    cabinet
  };

  assert.deepEqual(actual, expected, msg);
  assert.end();
});

test(`BassAmp`, assert => {
  const msg = `should have volume, lowCut, and cabinet`;
  const level = 2;
  const cabinet = `vintage`;

  const actual = BassAmp({
    lowCut: level,
    volume: level,
    cabinet
  });
  const expected = {
    lowCut: level,
    volume: level,
    cabinet
  };

  assert.deepEqual(actual, expected, msg);
  assert.end();
});

test(`ChannelStrip`, assert => {
  const msg = `should have inputLevel, lowCut, and volume`;
  const level = 2;

  const actual = ChannelStrip({
    inputLevel: level,
    lowCut: level,
    volume: level
  });
  const expected = {
    inputLevel: level,
    lowCut: level,
    volume: level
  };

  assert.deepEqual(actual, expected, msg);
  assert.end();
});

在CodePen运行

如果仔细看,您可能会发现我们在确定哪些对象具有哪些属性方面更加具体,因为有了组合,我们就可以做到。它不是类继承的真正选择。从类继承时,即使您不想要任何东西,也可以得到所有东西。
在这一点上,您可能会想自己:“很好,但是原型在哪里?”

要了解这一点,您必须了解有三种不同的原型OO。

JavaScript原型链

您可能已经开始意识到,串联继承是启用JavaScript中对象组合的秘诀,这使原型代理和函数继承都变得更加有趣。

当大多数人想到JavaScript中的原型OO时,他们就会想到原型代理。代理原型不是类继承的绝佳替代品-对象组合

为什么对象组合对脆弱的基类问题没有免疫力

要了解脆弱的基类问题以及为什么它不适用于合成,首先必须了解它是如何发生的:

C调用super,在B运行代码。B调用super这在运行的代码A

A和B包含C和D所需的无关功能。D是一个新的用例,在A的初始化代码中的行为与C所需要的行为略有不同。因此,新手开发人员开始调整A的初始化代码。C休息,因为它依赖于现有的行为,并D工作开始。

我们在这里拥有的是散布在A和B之间的功能,C和D需要以各种方式使用它们。CD不使用的所有功能AB ......他们只是想继承这已经在定义了一些东西AB。但是通过继承和调用super, 您就不必对继承的内容有所选择。您继承了所有内容:

面向对象语言的问题在于,它们具有随身携带的所有隐式环境。您想要香蕉,但是得到的是一只大猩猩,里面盛着香蕉和整个丛林。

使用组合Composition

想象一下您可以拥有功能而不是class:

feat1, feat2, feat3, feat4

C需要feat1feat3D需要feat1feat2feat4

const C = compose(feat1, feat3);
const D = compose(feat1, feat2, feat4);

现在,想象一下您发现D需要与feat1稍有不同的行为。实际上,它不需要更改feat1,而是可以制作一个自定义版本的feat1并使用它。您仍然可以继承“ feat2”和“ feat4 ”的现有行为,而无需进行任何更改:

const D = compose(custom1, feat2, feat4);

并且C不受影响。

类继承不可能实现的原因是,当您使用类继承时,您会购买整个现有的类分类法。

如果您想为新的用例做些改动,要么最终要复制现有分类法的某些部分(必要性重复问题),要么重构所有依赖于现有分类法的内容,以使分类法适应新的用途。由于脆弱的基层问题。

Composition 对两者都没有影响。

您认为您知道原型,但是…

如果您被教导构建类或构造函数并从中继承,那么您所教的不是原型继承。教会您如何使用原型模仿类继承。

在JavaScript中,类继承是基于很久以前内置在该语言中的非常丰富,灵活的原型继承功能之上的,但是当您使用类继承时-即使是在原型之上构建的ES6 + class继承,您也不是使用原型OO的全部功能和灵活性。

在JavaScript中使用类继承就像将新的Tesla Model S推向经销商并将其换成生锈的2011年的比亚迪F0一样。

参考

Master the JavaScript Interview: What’s the Difference Between Class & Prototypal Inheritance?

上一篇 下一篇

猜你喜欢

热点阅读