2011-01-31 14 views
5

Uczę się Clojure i potrzebuję popchnięcia we właściwym kierunku z tym problemem, który wymyśliłem.Pytanie idiomatycznego clojure o transformacji sekwencji

Mam sekwencję zdarzeń. Każde wydarzenie zawiera "datę".

(def events 
    [ 
    [1509 :marry "Catherine of Aragon"] 
    [1527 :unmarry "Catherine of Aragon"] 
    [1533 :marry "Anne Boleyn"] 
    [1536 :unmarry "Anne Boleyn"] 
    [1536 :marry "Jane Seymour"] 
    [1537 :unmarry "Jane Seymour"] 
    [1540 :marry "Anne of Cleves"] 
    [1540 :unmarry "Anne of Cleves"] 
    [1540 :marry "Catherine Howard"] 
    [1542 :unmarry "Catherine Howard"] 
    [1543 :marry "Catherine Parr"]]) 

Chcę przekonwertować to na leniwą oś czasu, tj. Sekwencję zawierającą jeden wektor na rok. Począwszy od roku pierwszego wydarzenia i kontynuowania nieskończoności.

[[[:marry "Catherine of Aragon"]] [] [] [] [] [] [] [] [] [] [] [] [] [] [] [] [] [] [[:unmarry "Catherine of Aragon"]] [] [] [] [] [] [[:marry "Ane Boleyn"]] [] [] [[:unmarry "Anne Boleyn"] [:marry "Jayne Seymour"]] ...] 
+0

Pomyślę ponownego formatowania do przestrzegania lisp konwencję/Clojure oddania końcowego znaku parens w jednej linii, patrz: http://techbehindtech.com/2010/12/09/clojure-good-coding-guidelines/ – 0x89

Odpowiedz

8
(def timeline 
    (let [events-by-year (group-by first events)] 
    (map #(map next (events-by-year %)) 
     (iterate inc (reduce min (keys events-by-year)))))) 

Szybki test:

=> (take 30 timeline) 
(((:marry "Catherine of Aragon"))()()()()()()()()()()()()()()()() 
() ((:unmarry "Catherine of Aragon"))()()()()() ((:marry "Anne Boleyn"))() 
() ((:unmarry "Anne Boleyn") (:marry "Jane Seymour")) ((:unmarry "Jane Seymour")) 
()) 
3

Proponuję coś takiego:

(defn timeline 
    ([] (timeline (ffirst *events*) *events*)) 
    ([time evts] 
    (let [[now later] (split-with #(= time (first %)) evts)] 
     (cons (map rest now) 
      (lazy-seq (timeline (inc time) later)))))) 

Test:

user> (take 30 (timeline)) 
(((:marry "Catherine of Aragon"))()()()()()()()()()()()()()()()()() 
((:unmarry "Catherine of Aragon"))()()()()() ((:marry "Anne Boleyn"))()() 
((:unmarry "Anne Boleyn") (:marry "Jane Seymour")) ((:unmarry "Jane Seymour"))()) 

jestem przy założeniu, że lista zdarzeń jest nieskończony, jak również :)

Zaktualizowany z usprawnieniami i zapożyczeniem kilku pomysłów z cgrand (dzięki :)

+0

Dzięki. Przerwa, gdy istnieje więcej niż jedno wydarzenie w tym samym dniu, tj. [1515: c] [1515: d]. Jednak jest tam wiele dobrych wskazówek, więc może sam mogę sam wypróbować resztę. – GHZ

+0

@GHZ Ah, tęskniłem za tym. Cgrand's i tak był milszy :) Refaktoryzowany, pozostawiając jako drugą opcję. –