美文网首页
clojure specter的简单使用

clojure specter的简单使用

作者: Tony_Yang | 来源:发表于2020-10-19 10:34 被阅读0次

    为什么要使用 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}]}]
    
    • 其他还有很多方法,可以去官网学习一下。。。

    相关文章

      网友评论

          本文标题:clojure specter的简单使用

          本文链接:https://www.haomeiwen.com/subject/oforwctx.html