草庐IT

javascript - 为什么Array.prototype.forEach无法链接?

coder 2024-05-07 原文

我今天了解到forEach()返回undefined。真是浪费!

如果它返回了原始数组,那么它将更灵活,而不会破坏任何现有代码。有什么原因forEach返回undefined

无论如何,是否可以将forEachmapfilter等其他方法链接起来?

例如:

var obj = someThing.keys()
.filter(someFilter)
.forEach(passToAnotherObject)
.map(transformKeys)
.reduce(reduction)

无法使用,因为forEach不想很好玩,要求您再次在forEach之前运行所有方法,以使对象处于forEach所需的状态。

最佳答案

您想要的通过method cascading被称为method chaining。简要描述它们:

  • 方法链接是一种方法返回一个对象,该对象具有您立即调用的另一个方法。例如,使用jQuery:
    $("#person")
        .slideDown("slow")
        .addClass("grouped")
        .css("margin-left", "11px");
    
  • 方法级联是在同一对象上调用多个方法时。例如,在某些语言中,您可以执行以下操作:
    foo
        ..bar()
        ..baz();
    

    这等效于JavaScript中的以下内容:
    foo.bar();
    foo.baz();
    

  • JavaScript没有用于方法级联的任何特殊语法。但是,如果第一个方法调用返回this,则可以使用方法链接来模拟方法级联。例如,在以下代码中,如果bar返回this(即foo),则链接等效于级联:
    foo
        .bar()
        .baz();
    

    诸如filtermap之类的某些方法可链接但不可级联,因为它们返回的是新数组,而不是原始数组。

    另一方面,forEach函数不可链接,因为它不返回新对象。现在,出现了一个问题,即forEach是否应该可级联。

    当前,forEach不可级联。但是,这并不是真正的问题,因为您可以简单地将中间数组的结果保存在变量中,并在以后使用:
    var arr = someThing.keys()
        .filter(someFilter);
    
    arr.forEach(passToAnotherObject);
    
    var obj = arr
        .map(transformKeys)
        .reduce(reduction);
    

    是的,此解决方案看起来比您所需的解决方案难看。但是,由于以下几个原因,我比您的代码更喜欢它:
  • 这是一致的,因为可链接方法未与可级联方法混合。因此,它促进了编程的功能风格(即无副作用的编程)。

    从本质上来说,级联是一种有效的操作,因为您正在调用方法而忽略结果。因此,您称该操作为副作用而不是结果。

    另一方面,像mapfilter这样的可链接函数没有任何副作用(如果它们的输入函数没有任何副作用)。它们仅用于结果。

    以我的拙见,将诸如mapfilter之类的可链接方法与诸如forEach(如果是可级联)的可级联函数混合在一起是令人讨厌的,因为这会在原本纯净的转换中带来副作用。
  • 这是明确的。正如The Zen of Python教给我们的那样,“显式胜于隐式”。方法级联只是语法糖。它是隐式的,并且要付出一定的代价。代价是复杂性。

    现在,您可能会说我的代码看起来比您的代码更复杂。如果是这样,您将通过封面来判断这本书。 Ben Moseley和Peter Marks在其著名的论文Out of the Tar Pit中描述了不同类型的软件复杂性。

    他们列表中的第二大软件复杂度是对控制流的明确关注引起的复杂度。例如:
    var obj = someThing.keys()
        .filter(someFilter)
        .forEach(passToAnotherObject)
        .map(transformKeys)
        .reduce(reduction);
    

    上面的程序明确涉及控制流,因为您明确指出.forEach(passToAnotherObject)应该在.map(transformKeys)之前发生,即使它对整体转换没有任何影响。

    实际上,您可以将其从等式中完全删除,并且没有任何区别:
    var obj = someThing.keys()
        .filter(someFilter)
        .map(transformKeys)
        .reduce(reduction);
    

    这表明.forEach(passToAnotherObject)首先没有关系任何事情。由于这是一种副作用操作,因此应与纯代码分开存放。

    当像我上面那样明确地编写它时,不仅可以将纯代码与副作用代码分开,而且可以选择何时评估每个计算。例如:
    var arr = someThing.keys()
        .filter(someFilter);
    
    var obj = arr
        .map(transformKeys)
        .reduce(reduction);
    
    arr.forEach(passToAnotherObject); // evaluate after pure computation
    

    是的,您仍然明确关心控制流程。但是,至少现在您知道.forEach(passToAnotherObject)与其他转换无关。

    因此,您已经消除了(但不是全部)由于明确关注控制流而导致的部分复杂性。

  • 出于这些原因,我相信forEach的当前实现实际上是有益的,因为由于明确关注控制流,它使您无法编写引入复杂性的代码。

    从我以前在BrowserStack工作时的亲身经历得知,对控制流的明确关注是大型软件应用程序中的一个大问题。这确实是一个现实世界的问题。

    编写复杂的代码很容易,因为复杂的代码通常是较短的(隐式)代码。因此,总是希望在纯计算过程中加入像forEach这样的副作用函数,因为它需要较少的代码重构。

    但是,从长远来看,它会使您的程序更加复杂。想一想,当您离开工作所在的公司而其他人必须维护您的代码时,几年后会发生什么。您的代码现在看起来像:
    var obj = someThing.keys()
        .filter(someFilter)
        .forEach(passToAnotherObject)
        .forEach(doSomething)
        .map(transformKeys)
        .forEach(doSomethingElse)
        .reduce(reduction);
    

    现在,阅读您的代码的人必须假设您链中所有其他的forEach方法都是必不可少的,进行额外的工作以了解每个函数的作用,自己弄清楚这些额外的forEach方法对于计算obj并不是必需的,请消除它们从她的代码思维模式开始,只专注于基本部分。

    这给程序增加了很多不必要的复杂性,并且您认为这使程序更简单。

    关于javascript - 为什么Array.prototype.forEach无法链接?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29228064/

    有关javascript - 为什么Array.prototype.forEach无法链接?的更多相关文章

    1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    2. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

      我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

    3. ruby - 在 Ruby 中实现 `call_user_func_array` - 2

      我怎样才能完成http://php.net/manual/en/function.call-user-func-array.php在ruby中?所以我可以这样做:classAppdeffoo(a,b)putsa+benddefbarargs=[1,2]App.send(:foo,args)#doesn'tworkApp.send(:foo,args[0],args[1])#doeswork,butdoesnotscaleendend 最佳答案 尝试分解数组App.send(:foo,*args)

    4. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    5. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

      我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

    6. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

      我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

    7. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

      为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

    8. ruby-on-rails - 无法使用 Rails 3.2 创建插件? - 2

      我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby​​1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在

    9. ruby - 无法运行 Rails 2.x 应用程序 - 2

      我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

    10. ruby-on-rails - 无法在centos上安装therubyracer(V8和GCC出错) - 2

      我正在尝试在我的centos服务器上安装therubyracer,但遇到了麻烦。$geminstalltherubyracerBuildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingtherubyracer:ERROR:Failedtobuildgemnativeextension./usr/local/rvm/rubies/ruby-1.9.3-p125/bin/rubyextconf.rbcheckingformain()in-lpthread...yescheckingforv8.h...no***e

    随机推荐