jjzjj

javascript - 将Fluture与Ramda一起使用

coder 2024-07-29 原文

我当时使用Bluebird进行异步处理,但现在必须进行大量的空/空/错误检查,如果不想使用其他路由,我也不想这么做。我想使用monad,但尚未完全了解它。

我也希望它能与ramda的pipe / compose一起很好地播放,因为我的大多数其他代码都整齐地封装在功能管道中。根据many discussions的说法,与Promises相比,首选Monadic Futures(似乎建议使用Fluture),并且在将来的版本中可能会删除对pipeP和composeP的支持。

Fluture似乎是一个不错的选择,因为它可以与遵循fantasy-land specs的库(如ramda)一起很好地运行。

但是,我完全迷失了如何实现将Ramda的管道与Fluture集成在一起的东西。我需要一些示例代码的帮助。

例如:

我有一个数据库调用,返回一个对象数组。该数组可以具有值,为空或未定义。我有一个功能管道,可以转换数据并将其返回到前端。

示例 promise 代码:

fancyDBCall1(constraints)
  .then(data => {
    if (!data || data.length === 0) {
      return []
    }
    return pipe(
    ...
    transformation functions
    ...
    )(data)
  })
  .then(res.ok)
  .catch(res.serverError) 

有人可以提出一些指导进行下一步工作的好方法吗?

最佳答案

你能做什么 ?

因此,什么时候可以使用您的代码。但首先,让我们谈谈Monads。

在此代码中,可以使用3种类型的Monad:

  • 也许(数据库可能返回一些信息,或nothing)
  • 任一(例如,如果某些数据验证失败)
  • Fluture(以替换 promise 。Fluture是与 promise 中的不同!)

  • 也许是这样,也许不是!

    让我们分解一下代码。我们要做的第一件事是确保fancyDBCall1(constraints)返回Maybe。这意味着它可能返回结果,或者什么都不返回。

    但是,您的fancyDBCall1是异步操作。这意味着它必须返回Future。这里的窍门不是让它返回值的 future ,例如Future <Array>使其返回Future < Maybe Array >

    哇,听起来很复杂!

    只是想像它而不是:Future.of('world');
    您具有:Future.of( Maybe( 'world' ) );
    还不错吧?

    这样,您可以避免在代码中执行空检查!以下行将消失:
    if (!data || data.length === 0) {
      return []
    }
    

    您的示例如下所示:
    /*
     * Accepts <Maybe Array>.
     * Most ramda.js functions are FL compatible, so this function
     * would probably remain unchanged.
     **/
    const tranform = pipe( .... ); 
    
    // fancyDBCall1 returns `Future <Maybe Array>`
    fancyDBCall1(constraints)
      .map( transform )
      .fork( always(res.serverError), always(res.ok) );
    

    看看我们的代码看起来有多好?但是,等等,还有更多!

    我们要么走得更远,要么我们不走!

    因此,如果您一直在密切注意,您就会知道我想念一些东西。当然,我们现在正在处理空检查,但是如果transform崩溃了怎么办?好吧,您将说“我们发送res.serverError”。

    好的。这还算公平。但是,例如,如果transform函数由于用户名无效而失败,该怎么办?

    您会说您的服务器炸毁了,但事实并非如此。您的异步查询很好,但是我们得到的数据却不正确。这是我们可以预料的,这不像 meteor 撞击我们的服务器场一样,只是有些用户给我们发送了一封无效的电子邮件,我们需要告诉他!

    这里的窍门是去改变我们的transform函数:
    /*
     * Accepts <Maybe Array>.
     * Returns <Maybe <Either String, Array> >
     **/
    const tranform = pipe( .... ); 
    

    哇,耶稣香蕉!这是什么黑魔法?

    在这里,我们说我们的转换可能不返回任何东西,或者可能返回一个Either。这要么是字符串(左分支始终是error),要么是值的数组(右分支总是正确的结果!)。

    全部放在一起

    所以,是的,这真是一次漫长的旅程,你不是说吗?为了给您一些具体的代码供您使用,这些结构的一些代码可能看起来像这样:

    首先,我们开始使用Future <Maybe Array>:
    const { Future } = require("fluture");
    const S = require("sanctuary");
    
    const transform = S.map(
      S.pipe( [ S.trim, S.toUpper ] )
    );
    
    const queryResult = Future.of( 
      S.Just( ["  heello", "  world!"] ) 
    );
    
    //const queryResult2 = Future.of( S.Nothing );
    
    const execute = 
      queryResult
        .map( S.map( transform ) )
        .fork(
          console.error,
          res => console.log( S.fromMaybe( [] ) ( res ) )
        );
    

    您可以玩queryResultqueryResult2。这应该使您对Maybe monad可以做什么有所了解。

    请注意,在这种情况下,我使用的是ramt的纯净版本Sanctuary,因为它是Maybe类型,但是您可以使用任何Maybe类型库并对此感到满意,代码的想法是相同的。

    现在,让我们添加Either。

    首先,让我们关注我们的转换函数,我对其进行了一些修改:
    const validateGreet = array =>
      array.includes("HELLO")       ?
      S.Right( array )    :
      S.Left( "Invalid Greeting!" );
    
    // Receives an array, and returns Either <String, Array>
    const transform = S.pipe( [
      S.map( S.pipe( [ S.trim, S.toUpper ] ) ),
      validateGreet
    ] );
    

    到目前为止,一切都很好。如果数组满足条件,则返回带有数组的Either的右分支,而不是带有错误的左分支。

    现在,让我们将其添加到前面的示例中,该示例将返回Future <Maybe <Either <String, Array>>>
    const { Future } = require("fluture");
    const S = require("sanctuary");
    
    const validateGreet = array =>
      array.includes("HELLO")       ?
      S.Right( array )    :
      S.Left( "Invalid Greeting!" );
    
    // Receives an array, and returns Either <String, Array>
    const transform = S.pipe( [
      S.map( S.pipe( [ S.trim, S.toUpper ] ) ),
      validateGreet
    ] );
    
    //Play with me!
    const queryResult = Future.of( 
      S.Just( ["  heello", "  world!"] ) 
    );
    
    //Play with me!
    //const queryResult = Future.of( S.Nothing );
    
    const execute = 
      queryResult
        .map( S.map( transform ) )
        .fork(
          err => {
              console.error(`The end is near!: ${err}`);
              process.exit(1);
          },
          res => {
            // fromMaybe: https://sanctuary.js.org/#fromMaybe
            const maybeResult = S.fromMaybe( S.Right([]) ) (res);
    
            //https://sanctuary.js.org/#either
            S.either( console.error ) (  console.log ) ( maybeResult )
          }
        );
    

    那么,这告诉我们什么呢?

    如果遇到异常(未预料到的异常),我们将打印The end is near!: ${err},然后干净地退出该应用程序。

    如果数据库未返回任何内容,则输出[]

    如果数据库确实返回了某些内容并且该内容无效,则我们输出"Invalid Greeting!"

    如果数据库返回了不错的东西,我们将其打印出来!

    耶稣香蕉,这很多!

    是的,是的。如果您从Maybe,Either和Flutures入手,则有很多概念需要学习,并且感到不知所措是正常的。

    我个人不知道Ramda的任何良好且活跃的Maybe/Either库,(也许您可以尝试Folktale的Maybe/Result类型?),这就是为什么我使用了Sanctuary,它是Ramda的一个克隆,它更纯净并且集成得很好与Fluture。

    但是,如果您需要从某个地方开始,则可以随时查看社区问题聊天并发布问题。阅读文档也有很大帮助。

    希望能帮助到你!

    关于javascript - 将Fluture与Ramda一起使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44879042/

    有关javascript - 将Fluture与Ramda一起使用的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

      我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

    3. 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

    4. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

      很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

    5. ruby - 在 Ruby 中使用匿名模块 - 2

      假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

    6. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

      我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

    7. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    8. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    9. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

      我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

    10. ruby - 在 64 位 Snow Leopard 上使用 rvm、postgres 9.0、ruby 1.9.2-p136 安装 pg gem 时出现问题 - 2

      我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po

    随机推荐