Clojure 中的高阶 if-then-else?

如何解决Clojure 中的高阶 if-then-else?

如果数据满足特定条件,我经常不得不通过函数运行我的数据。通常,函数 f 和条件检查器 pred 都被参数化为数据。出于这个原因,我发现自己希望得到一个既不知道 if-then-else 也不知道 f 的高阶 pred

例如,假设我想将 10 添加到 (range 5) 中的所有偶数。而不是

(map #(if (even? %) (+ % 10) %) (range 5))

我更喜欢有一个帮手——我们称之为fork——然后这样做:

(map (fork even? #(+ % 10)) (range 5))

我可以继续将 fork 实现为函数。它看起来像这样:

(defn fork
  ([pred thenf elsef]
   #(if (pred %) (thenf %) (elsef %)))
  ([pred thenf]
   (fork pred thenf identity)))

这可以通过优雅地组合 core 函数来完成吗?也许是一些不错的 juxt / apply / some 链?

或者,您知道实现上述(或类似)的任何 Clojure 库吗?

解决方法

正如 Alan Thompson 所提到的,如今,cond-> 是一种相当标准的方式,可以隐式地让“else”部分“返回不变的值”。不过,它并没有真正解决您成为高阶人士的希望。我还有另一个不喜欢 cond-> 的原因:我认为(并且在发明 cond-> 时争论过)它遍历每个匹配的测试是错误的,而不仅仅是第一个。无法将 cond-> 用作 cond 的类似物。

如果您同意我的观点,您可以尝试使用 flatland.useful.fn/fix 或该系列中的其他工具之一,这些工具是我们在 cond->1 之前编写的。

to-fix 正是您的 fork,除了它可以处理多个子句并接受常量和函数(例如,您可能想将 10 添加到其他偶数但将 0 替换为 20 ):

(map (to-fix zero? 20,even? #(+ % 10)) xs)

使用 cond-> 很容易复制 fix 的行为,但反过来不行,这就是为什么我认为 fix 是更好的设计选择。


1 显然,我们距离 fix 最终版本 10 周年只有几周的时间。时光飞逝。

,

我同意为此拥有某种高阶函数构造可能非常有用,但我不知道任何此类构造。确实,您可以实现更高阶的 fork 函数,但它的用处非常有限,并且可以使用 ifcond-> 宏轻松实现,如其他答案中所建议的.

然而,我想到的是transducers。您可以相当轻松地实现一个 forking 转换器,该转换器可以与其他转换器组合以构建强大而简洁的序列处理算法。

实现可能如下所示:

(defn forking [pred true-transducer false-transducer]
  (fn [step]
    (let [true-step (true-transducer step)
          false-step (false-transducer step)]
      (fn
        ([] (step))
        ([dst x] ((if (pred x) true-step false-step) dst x))
        ([dst] dst))))) ;; flushing not performed.

这就是您在示例中使用它的方式:

(eduction (forking even?
                   (map #(+ 10 %))
                   identity)

          (range 20))
;; => (10 1 12 3 14 5 16 7 18 9 20 11 22 13 24 15 26 17 28 19)

但它也可以comp与其他传感器一起构建更复杂的序列处理算法:

(into []

      (comp (forking even?
                     (comp (drop 4)
                           (map #(+ 10 %)))
                     (comp (filter #(< 10 %))
                           (map #(vector % % %))
                           cat))
            (partition-all 3))
      
      (range 20))
;; => [[18 20 11] [11 11 22] [13 13 13] [24 15 15] [15 26 17] [17 17 28] [19 19 19]]
,

根据细节,使用 cond-> 宏和朋友通常最容易实现此目标:

(let [myfn (fn [val] 
             (cond-> val
               (even? val) (+ val 10))) ]

结果

  (mapv myfn (range 5)) => [10 1 14 3 18]

有一个变体 in the Tupelo library 有时很有用:

(mapv #(cond-it-> %
         (even? it) (+ it 10))
  (range 5))

允许您使用特殊符号 it 将值传递到多个阶段。


如示例所示,您可以选择定义和命名转换器函数(我最喜欢的),或使用函数文字语法 #(...)

,

定义 fork(具有三个输入)的另一种方法可能是:

(defn fork [pred then else]
  (comp
    (partial apply apply)
    (juxt (comp {true then,false else} pred) list)))

请注意,在此版本中,输入和输出可以接收零个或多个参数。但是让我们采用更结构化的方法,定义一些其他有用的组合器。让我们从定义 pick 开始,它对应于态射的分类余积(和):

(defn pick [actions]
  (fn [[tag val]]
    ((actions tag) val)))

;alternatively
(defn pick [actions]
  (comp
    (partial apply apply)
    (juxt (comp actions first) rest)))

例如(mapv (pick [inc dec]) [[0 1] [1 1]]) 给出 [2 0]。使用 pick 我们可以定义 switch,它的作用类似于 case:

(defn switch [test actions]
  (comp
    (pick actions)
    (juxt test identity)))

例如(mapv (switch #(mod % 3) [inc dec -]) [3 4 5]) 给出 [4 3 -5]。使用 switch 我们可以轻松定义 fork:

(defn fork [pred then else]
  (switch pred {true then,false else}))

例如(mapv (fork even? inc dec) [0 1]) 给出 [1 0]。最后,使用 fork 让我们还定义 fork*,它接收零个或多个谓词和动作对,并像 cond 一样工作:

(defn fork* [& args]
  (->> args
       (partition 2)
       reverse
       (reduce
         (fn [else [pred then]]
           (fork pred then else))
         identity)))

;equivalently
(defn fork* [& args]
  (->> args
       (partition 2)
       (map (partial apply (partial partial fork)))
       (apply comp)
       (#(% identity))))

例如(mapv (fork* neg? -,even? inc) [-1 0 1]) 给出 [1 1 1]

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-