【问题标题】:clojure, how to reduce repeated code (potentially using macro)clojure,如何减少重复代码(可能使用宏)
【发布时间】:2021-08-12 01:45:08
【问题描述】:

TL;DR

如何减少重复代码以下,例如从job-inventory 创建两个作业/触发器,而不是重复两次并创建术语


;; deps in project.clj
;; [clojurewerkz/quartzite "2.1.0"]


(ns hello.scheduler
  (:require [clojurewerkz.quartzite.scheduler :as qs]
            [clojurewerkz.quartzite.triggers :as t]
            [clojurewerkz.quartzite.jobs :as j]
            [clojurewerkz.quartzite.jobs :refer [defjob]]
            [clojurewerkz.quartzite.schedule.cron :as cron])
  (:use clojure.tools.logging)
  (:gen-class))

(def job-inventory
  [{:name "add" :task '(+ 1 1) :cron "0/5 * * ? * *"}
   {:name "multiply" :task '(* 4 5)  :cron "0/3 * * ? * *"}])

(defjob add [ctx] (info "add called, return" (+ 1 1)))
(defjob multiply [ctx] (info "multiply called, return" (* 2 3)))

(defn auto
  []
  (let [s   (-> (qs/initialize) qs/start)
        _ (qs/clear! s)
        job (j/build
             (j/of-type add)
             (j/with-identity (j/key "job.add")))
        trigger (t/build
                 (t/with-identity (t/key "trigger.add"))
                 (t/start-now)
                 (t/with-schedule (cron/schedule
                                   (cron/cron-schedule "0/5 * * ? * *"))))
        _ (qs/schedule s job trigger)
        
        job (j/build
             (j/of-type multiply)
             (j/with-identity (j/key "job.multiply")))
        trigger (t/build
                 (t/with-identity (t/key "trigger.multiply"))
                 (t/start-now)
                 (t/with-schedule (cron/schedule
                                   (cron/cron-schedule "0/3 * * ? * *"))))
        _ (qs/schedule s job trigger)
        ]
    ))


类似于http://clojurequartz.info/articles/getting_started.html 中描述的内容, 我有代码块来创建作业并将它们挂钩以执行

问题是,当我得到越来越多的它们时,我想知道是否可以有更好的方法来管理它们,比如从 job-inventory 创建/生成,而不是实际创建像 add 或 @987654327 这样的变量@

所以,要求再循环一层 有没有办法利用函数编程,并避免创建新名称(在传统语言中说 python qt,如果我有一组按钮,我可以粉碎成一个巨大的字典,然后循环创建/禁用,而不是实际创建每个命名为顶级变量)

我试过宏,但它说无法解析类添加,所以我猜我用错了

【问题讨论】:

  • 您可能希望考虑的相关库:github.com/overtone/at-at
  • @AlanThompson 感谢您的关注,只有三个任务,但在这样的石英术语中,我必须构建一个类型为 job-a 的 job-a 和一个 trigger-a,然后hook up schedule,可以考虑看看他们的教程,这些job之间没有严格的联系或依赖。
  • @AlanThompson 感谢您提供的线索,肯定会看一下,这里我有一个项目已经配置了 cron 格式计划,所以如果兼容会更有吸引力,感谢您的时间:)
  • @AlanThompson 这里的问题可能是,我怎么能在循环中调用defjob job-a defjob job-b defjob job-c(然后引用它们),一种代码生成技巧。
  • @AlanThompson 嗨艾伦,我已经减少了描述,只显示代码片段,希望它稍微更清楚

标签: functional-programming clojure cron quartzite


【解决方案1】:

要记住的关键是函数就是数据。虽然您不能非常轻松地动态创建类型(与通过reify 实现接口的实例相反),但您可以静态创建一个类,然后代理您的函数。

首先让我们将job-inventory:task 设为一个函数。

(def job-inventory
  [{:name "add" :task (fn [] (println (+ 1 1))) :cron "0/5 * * ? * *"}
   {:name "multiply" :task (fn [] (println (* 4 5)))  :cron "0/3 * * ? * *"}])

然后我们需要代理作业类。这会执行它在作业数据中找到的函数。

;; require clojurewerkz.quartzite.conversion :as qc
(defjob proxy-job [ctx]
  (let [ctx (qc/from-job-data ctx)]
    ((ctx "proxied-fn"))))

然后我们创建一个调度函数,它从作业库存中获取一个映射,并使用代理作业将其调度到间接。

(defn schedule [scheduler {:keys [:name :task :cron]}]
  (let [job (j/build
             (j/of-type proxy-job)
             (j/using-job-data {"proxied-fn" task})
             (j/with-identity (j/key (str "job." name))))
        trigger (t/build
                 (t/with-identity (t/key (str "trigger." name)))
                 (t/start-now)
                 (t/with-schedule (cron/schedule
                                   (cron/cron-schedule "0/3 * * ? * *"))))]
  (qs/schedule scheduler job trigger)
  scheduler)

(reduce scheduler schedule job-inventory)

如果石英决定序列化/反序列化作业数据,这种方法可能会失败。对于感兴趣的读者,我将使用另一层间接来解决这个问题,作为一个简单的练习 - 老实说,我最初的想法是我们创建命名函数,然后我们通过符号引用,但它使代理复杂化,让你想知道为什么不只是使用defjob。如果您准备取消您的功能,您仍然可以在工作清单中引用它们并拥有基于数据的工作构建器功能。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-11-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多