【译】3 种不同的原型继承: ES6+ 版本

发表于 5年以前  | 总阅读数:1996 次

原文:http://www.zcfy.cc/article/425

这篇文章改编自 《Programming JavaScript Applications》 相关章节,我在这基础上扩充了一些内容,并新增了 ES6 部分。

为了更好地发挥 JavaScript 的能力,很重要的一点是理解 JavaScript 的 原生继承能力。这是一个在 JavaScript 实践和学习中经常被忽视的领域,但是理解了它能够获得极大的能力。

JavaScript 是表现力极强的编程语言之一。其中一个最好的特性是能够在没有类和类继承的情况下创建和继承对象。

结合代理原型(delegate prototypes)、运行时对象扩展和闭包,能够让你用三种不同的原型表达方式实现继承,它们与类继承方式相比有着显著的优点。

要想了解关于类继承的更多弊端,可以观看视频 "类继承已经过时了:怎样用基于原型的面向对象思维思考":(要翻墙----译者注)

代理 / 不同的继承方式

代理原型是一个对象,作为另外一个对象的基类。当你继承一个代理原型,新对象得到一个代理原型对象的引用。

当你要访问新对象的一个属性时,它先检查当前对象自己的属性,如果没有找到,它检查 [[Prototype]],然后继续沿着原型链往上查找,直到它到达 Object.prototypeObject.prototype 是大多数对象的根代理(root delegate)。

方法代理(method delegation)可以节省内存资源,因为你只需要为每个方法准备一份拷贝,它通过原型被所有的实例共享。有许多办法可以做到这一点,在 ES6 中可能看起来像下面这样:

class Greeter {
      constructor (name) {
        this.name = name || "John Doe";
      }
      hello () {
        return `Hello, my name is ${ this.name }`;
      }
    }

    const george = new Greeter("George");

    const msg = george.hello();

    console.log(msg); // Hello, my name is George

由于传统继承和类的扩展有许多问题,我不推荐使用这个技术。我将它展示在这里只是因为它与传统面向对象的模式类似,所以大家对这种方式可能比较熟悉。

你可能对用 ES5 构造函数的方式也比较熟悉:

function Greeter (name) {
      this.name = name || "John Doe";
    }

    Greeter.prototype.hello = function hello () {
      return "Hello, my name is " + this.name;
    }

    var george = new Greeter("George");

    var msg = george.hello();

    console.log(msg); // Hello, my name is George

我更喜欢使用 Object.create() 实现工厂函数(factory function)的方式。(在 JavaScript 中,任何函数都可以用来创建对象。当它不是一个构造函数时,它被叫做工厂函数):

const proto = {
      hello () {
        return `Hello, my name is ${ this.name }`;
      }
    };

    const greeter = (name) => Object.assign(Object.create(proto), {
      name
    });

    const george = greeter("george");

    const msg = george.hello();

    console.log(msg);

如果你不需要代理属性,你可以用 Object.create(null) 把 prototype 设置成 null

使用原型来代理的主要缺点是它不好存储状态。如果你要用对象或者存储来存储状态,改变数组或对象的任何一个成员会改变同时每一个实例上的内容,因为它保存在原型上。为了解决这个问题,你需要将它拷贝到每个对象上。

拼接继承 / 克隆 / 混合

拼接继承是一个从一个对象向另一个对象拷贝属性的过程,它不保持两个对象引用同一个对象。它依赖于 JavaScript 的动态对象扩展特性。

克隆一个非常好的保存对象默认状态的方式:ES6 中它可以通过 Obejct.assign() 实现,这个方法和 Lodash、Underscore 以及 jQuery 的 extend() 方法类似。

const proto = {
      hello: function hello() {
        return `Hello, my name is ${ this.name }`;
      }
    };

    const george = Object.assign({}, proto, {name: "George"});

    const msg = george.hello();

    console.log(msg); // Hello, my name is George

这个形式在混合模式(mixins)中也很常见。例如,你可以通过混合 EventEmitter3 的原型将任意一个对象变成事件派发器:

import Events from "eventemitter3";

    const object = {};

    Object.assign(object, Events.prototype);

    object.on("event", payload => console.log(payload));

    object.emit("event", "some data"); // "some data"

我们可以使用它来创建一个 Backbone 风格的事件派发模型:

import Events from "eventemitter3";

    const modelMixin = Object.assign({
      attrs: {},
      set (name, value) {
        this.attrs[name] = value;

        this.emit("change", {
          prop: name,
          value: value
        });
      },

      get (name) {
        return this.attrs[name];
      }
    }, Events.prototype);


    const george = { name: "george" };
    const model = Object.assign(george, modelMixin);

    model.on("change", data => console.log(data));

    model.set("name", "Sam");
    /*
    {
      prop: "name",
      value: "Sam"
    }
    */

拼接继承非常强大,但你还可以结合闭包将它变得更强大。

函数式继承

别将它和函数式编程混为一谈。

在 《JavaScript: The Good Parts》 中,Douglas Crockford 创造了函数式继承。函数式继承使用一个工厂函数,然后使用拼接继承将新属性添加进来。

如果一个函数创建的目的是扩展已有对象,这个模式通常被称为函数式混合。使用函数来扩展的主要好处是它能让你使用函数闭包来封装私有数据。换句话说,你可以执行私有状态。

将私有属性挂载在公有属性上有点小尴尬,用户不用通过调用合适的函数就能读写它们。我们确实想要将私有属性通过闭包(可以进一步了解"什么是闭包")对用户隐藏起来。看下面这个例子:

import Events from "eventemitter3";

    const rawMixin = function () {
      const attrs = {};

      return Object.assign(this, {
        set (name, value) {
          attrs[name] = value;

          this.emit("change", {
            prop: name,
            value: value
          });
        },

        get (name) {
          return attrs[name];
        }
      }, Events.prototype);
    };

    const mixinModel = (target) => rawMixin.call(target);

    const george = { name: "george" };
    const model = mixinModel(george);

    model.on("change", data => console.log(data));

    model.set("name", "Sam");
    /*
    {
      prop: "name",
      value: "Sam"
    }
    */

通过将 attrs 从一个公有属性变成一个私有变量,我们从公共 API 上移除了所有 attrs 相关的内容。唯一可以使用它的方式只能是通过特权方法,即那些定义在闭包的函数作用域中的方法,这些方法可以访问到私有数据。

注意上面的例子,我们使用 mixinModule() 包装实际的函数式混合 rawMixin()。这么做的原因是我们需要将 this 设置到函数中,我们通过 Function.prototype.call() 来做到这一点。我们可以不用这个 wrapper,将 this 通过参数直接传进去,但我不喜欢那样做。

组合模式好于类继承

"对象组合好于类继承" ~ 四人组, 《Design Patterns: Elements of Reusable Object Oriented Software》(《Design Patterns: Elements of Reusable Object-Oriented Software》,由 Erich Gamma、Richard Helm、Ralph Johnson 和 John Vlissides 合著(Addison-Wesley,1995)。这几位作者常被称为"四人组(Gang of Four)"。----译者注)

传统继承通过限制性分类创建了"归属(is-a)"关系,在新型用例里面,这种描述关系通常最终不适用。事实证明,我们经常采用"有一个(has-a)"、"使用一个(uses-a)" 或者 "能做(can-do)"关系。

组合模式更像一个全能的吉他手。无论想要什么样的效果都没问题,只需要将对应的功能插入进来。

const effect = compose(delay, distortion, robovoice); // Rock on!

什么时候你想要使用类继承?对于我来说,答案很简单:"从不使用。"

组合模式:

  • 更简单
  • 表现力更强
  • 更灵活

观看视频 "Composition Over Inheritance" 了解更多信息。(也要翻墙----译者注)

Stamps

Stamps 是可组合的函数工厂(composable factory functions)。我创建了 stamps 作为一个实验性的示例用在我的书 《Programming JavaScript Applications》 中。一切开始于以下有趣的挑战:

一种模拟 JavaScript 类的语法糖(我强烈不鼓励使用 JavaScript 类)。如果我们为 JavaScript 基于原型的面向对象创建类语法糖,并支持所有好的特性,那么它看起来是怎样的?

探索的结果得到了目前据我所知最受欢迎的 JavaScript 原型继承库:Stampit。

stampit 的设计被很多人认同,没过多久,它的衍生库开始出现。特别是,Tim Routowicz 写了杰出的 react-stamp 库用于可组合的 React 组件。我们需要一个标准让我们的 API 实现能够彼此一致。

于是我们有了 Stamp 规范。

一个 stamp 是一个可组合的工厂函数,它基于它的描述符(descriptor)返回对象实例。

stamp(options?: Object, ...args?: [...Any]) => instance: Object

每个 stamp 有一个方法叫做 compose()

Stamp.compose(...args?: [...Composable]) => Stamp

compose() 方法被调用,它以当前 stamp 为基础构造一个新的 stamp,一组可被组合元素作为参数被传入作为被组合元素列表被组合起来:

const combinedStamp = baseStamp.compose(composable1, composable2, composable3);

一个可被组合元素是一个 stamp 或者一个 POJO(Plain Old JavaScript Object) stamp 描述符。

compose() 还可以代表 stamp 的描述符。换句话说,描述符的,属性被附加到 stamp 的 .compose() 方法上,比如 stamp.compose.methods

描述符

可组合描述符(或者简称描述符)是一个元素数对象,它包含创建一个对象实例的必要信息。

一个描述符包含:

  • methods - 将被添加到对象原型上的方法集。
  • properties - 将被添加到对象原型上的属性集。
  • initializers - 一个函数数组,它会按顺序执行。Stamp 的细节和参数会被依次传给这些函数。
  • staticProperties - 将被拷贝和添加到 stamp 本身之上的静态属性集。

描述符还有其他一些属性,但是它们不是最常用的关键属性。你可以查看 descriptor 规范来进一步了解全貌。

为什么用 stamp?

原型继承很棒,JavaScript 的能力给了我们非常强大的工具去探索它的更多用法,但是它可以被约束得更加方便实用。

一些基本问题例如"我怎样通过一个授权方法访问私有数据?"以及"实现多重继承有哪些好的选择?"是许多 JavaScript 使用者都遇到过的难题。

让我们使用 stamp-utils 库中的 init()compose() 来回答所有这些问题。

  • compose(...composables:[...Composable]) => Stamp 接收任意个 composeables 参数并返回一个新的 stamp。
  • **init(...functions:[...Function]) => Stamp 接收任意个 initializer 函数并返回一个新的 stamp。

首先,我们会使用一个闭包来创建数据的私有空间:

const a = init(function () {
      const a = "a";

      Object.assign(this, {
        getA () {
          return a;
        }
      });
    });

    console.log(typeof a()); // "object"
    console.log(a().getA()); // "a"

它使用函数作用域来封装私有数据。注意 getter 必须定义在函数中,以便于访问到闭包内的变量。

这是另一个例子:

const b = init(function () {
      const a = "b";

      Object.assign(this, {
        getB () {
          return a;
        }
      });
    });

getB 中返回 a 不是我写错了,我故意这样写来说明两个 init 的私有变量叫同样的名也可以,getter 名字不一样就行,compose 的时候彼此不冲突:

const c = compose(a, b);

    const foo = c();
    console.log(foo.getA()); // "a"
    console.log(foo.getB()); // "b"

很好玩吧?上面同时继承了两个从各自私有域访问私有属性的授权方法。

但它简单得有点无聊,让我们看看还有什么玩法:

import { compose, init } from "stamp-utils";

    // 更多授权方法,操作一些私有数据。
    const availability = init(function () {
      let isOpen = false; // private

      Object.assign(this, {
        open () {
          isOpen = true;
          return this;
        },
        close () {
          isOpen = false;
          return this;
        },
        isOpen () {
          return isOpen;
        }
      });
    });

    //这是一个 stamp,它包含一些公有方法,和一些状态:
    const membership = compose({
      methods: {
        add (member) {
          this.members[member.name] = member;
          return this;
        },
        getMember (name) {
          return this.members[name];
        }
      },
      properties: {
        members: {}
      }
    });

    // 让我们设置一些默认值:
    const defaults = compose({
      properties: {
        name: "The Saloon",
        specials: "Whisky, Gin, Tequila"
      }
    });

    const overrides = init(function (overrides) {
      Object.assign(this, overrides);
    });

    // 传统继承实现不了下面这个模式。这个模式没有父类/子类耦合。
    // 不用一层一层深深嵌套来做多重继承
    // 代码良好、干净和可重用
    const bar = compose(availability, membership, defaults, overrides);
    const myBar = bar({name: "Moe\"s"});

    // 看起来有点笨,但保证了一切是一切该有的样子。
    const result = myBar.add({name: "Homer"}).open().getMember("Homer");
    console.log(result); // { name: "Homer" }
    console.log(`
      name: ${ myBar.name }
      isOpen: ${ myBar.isOpen() }
      specials: ${ myBar.specials }
    `);
    /*
      name: Moe"s
      isOpen: true
      specials: Whisky, Gin, Tequila
    */

class 怎么样?

如你所见,JavaScript 提供了一个非常有弹性的对象系统,它不需要依赖 class。那么为什么我们要添加 class 到我们的首选项里?因为许多人对其他语言中的 class 范式更熟悉,所以人们总是不断尝试在 JavaScript 中去模拟它。

继承在 JavaScript 中实现起来如此容易,容易到让那些想要实现它的人感到迷惑。为了让它变得难一些,我们添加了 class

好几个著名的 JavaScript 框架实现了模仿传统继承的机制,它们通过原型链来模拟类继承。添加一个官方的 class 关键字提供了一个规范的方法去实现类继承,但是依我的观点,你应该避免使用它。在 JavaScript 中,组合模式比类继承更简单,表现力更强,而且更加灵活。我没想到在什么使用场景下 class 能比原生的原型模式表现得更好。

几年来,我向人们发起挑战,看谁能提供一个使用场景,说明在这个场景下必须用类继承,然而目前为止还没看到有什么场景用类继承好,相反,我听到一大堆 常见误解。

结论

一旦你开始考虑不用 class 创建对象,继承采用原型,通过级联和 stamp 组合对象,你将会惊叹于 JavaScript 对象系统竟能变得如此简单、功能强大和灵活。

最近 stamp 社区成长了许多。我们已经创建了很多基于 stamp 的库和工具类。最被广泛使用的包括 Stampit (Stamp 的原始库)、 react-stamp 以及 stamp-utils。

这篇文章的所有例子都使用 stamp utils 编写。它提供了一个 compose() 方法的最小实现,附带一个 init() 函数,它是 compose 的一个语法糖:

const init = (...functions) => compose({ initializers: [...functions] });

下一次你需要创建有状态的对象产生许多实例,可以尝试使用 stamps。

英文原文:https://medium.com/javascript-scene/3-different-kinds-of-prototypal-inheritance-es6-edition-32d777fa16c9#.q4929515z

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:1年以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:1年以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:1年以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:1年以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:1年以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:1年以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:1年以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:1年以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:1年以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:1年以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:1年以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:1年以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:1年以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:1年以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:1年以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:1年以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:1年以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:1年以前  |  398次阅读  |  详细内容 »
 相关文章
为Electron程序添加运行时日志 5年以前  |  20424次阅读
Node.js下通过配置host访问URL 5年以前  |  5915次阅读
用 esbuild 让你的构建压缩性能翻倍 4年以前  |  5819次阅读
 目录