clojure specter的简单使用

为什么要使用 specter

为了说明为什么要推荐大家使用 specter,先举个实际用到的例子🌰

(def db {:custom {:choices {:current-id "1002"
                            :datas [{:category_id "1002"
                                     :sample [{:id "4"
                                               :name "不需要试穿"
                                               :default_flag "1"}
                                              {:id "5"
                                               :name "半成品试穿"
                                               :default_flag "0"}]}]}}})

上面的这个多层嵌套的数据结构在开发中是比较常见的,smaple是服装的试样列表,用户可以切换选择。

于是切换事件的代码是这样的:

;; 切换试样
(kf/reg-event-db
 :custom/change-sample
 (fn [db [id]]
   (let [datas (get-in db [:custom :choices :datas])
         cat-id (get-in db [:custom :choices :current-id])
         new-datas (map (fn [cat]
                          (if (= cat-id (:category_id cat))
                            (let [samples (:sample cat)
                                  new-sample (->> samples
                                                  (map (fn [sample]
                                                         (if (= "1" (:default_flag sample))
                                                           (assoc sample :default_flag "0")
                                                           sample)) )
                                                  (map (fn [sample]
                                                         (if (= id (:id sample))
                                                           (assoc sample :default_flag "1")
                                                           sample)) ))]
                              (assoc cat :sample new-sample))
                            cat)) datas)]
     (assoc-in db [:custom :choices :datas] new-datas))))

水平有限,实现方式有些小白... 总之代码一大堆,其实只做了两件事:

  • 当点击切换试样时,就把之前选中的试样的default_flag标记为0
  • 新选中的标记为1。

下面我再用specter实现一下:

;; 切换试样
(kf/reg-event-db
 :custom/change-sample
 (fn [db [id]]
   (let [cat-id (get-in db [:custom :choices :current-id])
         cur-cat-path [:custom :choices :datas s/ALL #(= cat-id (:category_id %))]
         pre-sel-path (into cur-cat-path
                            [:sample s/ALL #(= "1" (:default_flag %))])
         cur-sel-path (into cur-cat-path
                            [:sample s/ALL #(= id (:id %))])]
     (->> db
          (s/transform pre-sel-path #(assoc % :default_flag "0"))
          (s/transform cur-sel-path #(assoc % :default_flag "1"))))))

通过对比很容易发现,刚才这种方式对数据的处理更为简单直接,代码易读性更高😄

如何使用

  • 引用

    [com.rpl.specter :as s]
    
  • transform (transform apath transform-fn structure)

    使用起来非常简单,以刚才上面的样例举例说明:

    ① 第一个参数是路径的vector,里边存放的是每层的key

    [:custom :choices :datas s/ALL 当前选中的分类 :sample s/ALL 当前选中的试样]

注意的点:

  • s/ALL 它的意思是取当前数组的所有值
  • 从ALL获取的vector中找到当前选中的分类,可以使用条件函数:#(= cat-id (:category_id %))
  • 获取当前选中的试样,同理~

② 第二个参数是要执行的转换方法,该方法的参数是根据刚才的路径查找到的数据

​ 比如根据刚才的路径在该方法中的参数得到的就是当前选中的试样

​ 之后就是在方法中对该数据的处理,如#(assoc % :default_flag "1")

③ 第三个参数就是你最外层的那个数据 (刚才的就是db)

​ 如果是从当前的分类开始查找,那么这个参数就是当前选中的分类 current-category

​ 当然对应的路径应改为:[:sample s/ALL 当前选中的试样]

  • setvalue (setval apath aval structure)

    该方法跟transform用法比较类似,可以看下刚才的代码用setvalue的实现方式:

    ;; 切换试样
    (kf/reg-event-db
     :custom/change-sample
     (fn [db [id]]
       (let [cat-id (get-in db [:custom :choices :current-id])
             cur-cat-path [:custom :choices :datas s/ALL #(= cur-id (:category_id %))]
             pre-sel-path (into cur-cat-path
                                [:sample s/ALL #(= "1" (:default_flag %)) :default_flag])
             cur-sel-path (into cur-cat-path
                                [:sample s/ALL #(= id (:id %)) :default_flag])]
         (->> db
              (s/setval pre-sel-path "0")
              (s/setval cur-sel-path "1")))))
    

通过对比可以发现:

  1. 路径比transform的多了一个key :default_flag,也就是说它具体定位到了该对象的这个属性
  2. 函数调用时直接通过path设置了值,没有使用方法

总结:

如果对求得的值需要通过方法对其处理,建议使用transform;

如果只是单纯的对值进行赋值操作,建议使用setvalue。

  • select (select apath structure)

    返回一个vector,里边是查找到的所有元素

    如果未找到,则返回 []

    (def sample-path [:custom :choices :datas s/ALL #(= "1002" (:category_id %))
                      :sample s/ALL #(= "5" (:id %))])
    
    (s/select cur-sel-path db)
    ;; ->>>> [{:id "5" :name "半成品试穿" :default_flag "0"}]
    
  • select-first (select-first apath structure)

    返回查询到的第一个元素

    注意:如果未找到元素,则返回:nil

  • select-any (select-any apath structure)

    返回查询到的第一个元素

    注意:如果未找到元素,则返回:com.rpl.specter/NONE

  • select-any? (selected-any? apath structure)

    返回一个布尔值 true / false

总结

需要返回一个vector,使用select

需要直接返回一个元素,建议使用select-first

需要返回布尔值,即判断该元素是否存在,使用select-any?

  • nthpath (nthpath & indices)

指定下标

;; 查询指定下标的元素
(select [(nthpath 2)] [1 2 3])
;; => [3]

;; 注意可以指定多个下标参数
(select [(nthpath 0 0)] [[0 1 2] 2 3])
;; => [0]
  • filterer (filterer & path)

过滤sequence

;; 普通写法
(s/select [s/ALL even?] (range 10))
;; => [0 2 4 6 8]

;; filterer写法
(s/select-one (s/filterer even?) (range 10))
;; => [0 2 4 6 8]
  • view (view afn)

获取前面的结果并进行方法处理

;; 找出偶数并加一
(s/select [s/ALL even? (s/view inc)] [1 2 5 6 8])
;; => [3 7 9]

;; 找出偶数先加一再乘以10
(s/select [(s/filterer even?) s/ALL (s/view inc) (s/view #(* 10 %))]
          [1 2 5 6 8])
;; => [30 70 90]
  • walker (walker afn)

;; 找出偶数
(s/select (s/walker #(and (number? %) (even? %)))
          '(1 (3 4) 2 (6)))
;; => [4 2 6]

;; 注意(2 (3 4) 5 (6 7))的个数是偶数个,符合条件,所以不再向下寻找
(s/setval (s/walker #(and (counted? %) (even? (count %))))
          :double
          '(1
            (2 (3 4) 5 (6 7))
            (8 9)))
;; => (1 :double :double)

;; 注意(2 (3 4) 5)的个数是奇数个,不合符条件,所以继续再向下寻找(3 4)
(s/setval (s/walker #(and (counted? %) (even? (count %))))
          :double
          '(1
            (2 (3 4) 5)
            (8 9)))
;; => (1 (2 :double 5) :double)
  • selected? (selected? & path)

(s/setval [s/ALL
           (s/selected? (s/filterer even?)
                        (s/view count) #(> % 2))
           s/FIRST]
          "😆😆"
          [[6 7 8] [1 2 3] [5 6 7 8 9 10]])
;; => [[6 7 8] [1 2 3] ["😆😆" 6 7 8 9 10]]
  • collect (collect & paths)

根据给定的路径进行select求值,并把结果放到一个vector中

;; 将第一个元素值更新为它与其他所有偶数的和
(s/transform [(s/collect s/ALL even?) s/FIRST]
             (fn [evens first]
               (prn evens first) ;; => [4 6] 3
               (reduce + first evens))
             [3 4 5 6])
;; => [13 4 5 6]
  • collect-one (collect & paths)

同collect,只不过会把结果单独返回(而不是放到vector中)

;; 将最后一个元素值更新为它与第一个元素的和
(s/transform [(s/collect-one s/FIRST) s/LAST]
             (fn [first last]
               (prn first last) ;; => 1 3
               (+ first last))
             [1 2 3])
;; => [1 2 4]

注意:无论是collect,还是collect-one,它们都可以在已有的路径后面多次使用。如果使用了transform方法,那么collected得到的value们会作为参数按照先后顺序传递到tranform的处理方法中,并且根据transform的path最终对应的值会作为最后一个参数传递进去。

;; 多个collect的示例
(s/transform [s/ALL (s/collect-one :rate) (s/collect-one :deduct)
               :top s/ALL (s/collect-one :b) :a even?]
              (fn [rate deduct b e]
                (if b
                  100
                  (- (* rate e) deduct)))
              [{:rate 2.0
                :deduct 20
                :top [{:a 0 :b 1} {:a 1} {:a 2} {:a 3}]}
               {:rate 3.0
                :deduct 10
                :top [{:a 0} {:a 1} {:a 2} {:a 3}]}])
;; =>
[{:rate 2
  :deduct 20
  :top [{:a 100 :b 1} {:a 1} {:a -16} {:a 3}]}
 {:rate 3
  :deduct 10
  :top [{:a -10} {:a 1} {:a -4} {:a 3}]}]
  • 其他还有很多方法,可以去官网学习一下。。。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352