This page is not created by, affiliated with, or supported by Slack Technologies, Inc.
2018-09-14
Channels
- # 100-days-of-code (4)
- # announcements (1)
- # beginners (63)
- # boot (22)
- # braveandtrue (104)
- # calva (3)
- # cider (12)
- # cljs-dev (53)
- # cljsjs (3)
- # cljsrn (1)
- # clojure (180)
- # clojure-dev (14)
- # clojure-italy (4)
- # clojure-nl (11)
- # clojure-spec (15)
- # clojure-uk (60)
- # clojure-ukraine (1)
- # clojurescript (118)
- # clojutre (3)
- # core-async (12)
- # core-logic (17)
- # cursive (19)
- # datomic (45)
- # devcards (4)
- # emacs (7)
- # figwheel-main (218)
- # fulcro (27)
- # funcool (3)
- # graphql (1)
- # jobs (4)
- # leiningen (57)
- # off-topic (71)
- # pedestal (2)
- # portkey (17)
- # re-frame (5)
- # reitit (4)
- # remote-jobs (2)
- # ring (11)
- # rum (2)
- # shadow-cljs (14)
- # specter (11)
- # sql (34)
- # tools-deps (23)
Hi @manutter51, can you give me some pointers with implementing update-in and assoc-in?
I can try
There’s probably a nice recursive function for that
(def character
{:name "Smooches McCutes"
:attributes {:intelligence 10
:strength 4
:dexterity 5}})
(assoc-in character
[:attributes :intelligence] 19)
; --> {:name "Smooches McCutes",
:attributes {:intelligence 19, :strength 4, :dexterity 5}}
(update-in character
[:attributes :intelligence] + 19)
; --> {:name "Smooches McCutes",
:attributes {:intelligence 29, :strength 4, :dexterity 5}}
; and
(assoc (:attributes character) :intelligence 26)
; --> {:intelligence 26, :strength 4, :dexterity 5}
; stuck here...
;(assoc (assoc :intelligence 26)
but first, I want to write it manually for (assoc-in character [:attributes :intelligence] 19)
with just assoc
so let’s see…
character
is a map, and inside that map is another map, under the key :attributes
and inside the inner map is a key called :intelligence
that you want to change.
(booting up a sandbox repl, just a sec)
I’m here …
(assoc character :attributes (assoc (:attributes character) :intelligence 26))
; --> {:name "Smooches McCutes",
:attributes {:intelligence 26, :strength 4, :dexterity 5}}
Cool,
(def nested-map {:name "bob"
:attributes
{:strengh 10
:dexterity 5
:intelligence {:spatial 50
:verbal 20
:math 53}}})
lol, where have I seen that before 😄
(assoc nested-map :attributes (assoc (:attributes nested-map) :dexterity 100))
; --> {:name "bob",
:attributes {:intelligence {:spatial 50, :verbal 20, :math 53},
:dexterity 100,
:strengh 10}}
(assoc nested-map :attributes (assoc (:attributes nested-map) :intelligence (assoc (:intelligence (:attributes nested-map) ) :spatial 50 ) ) ; doesn't work,... I'm lost in here
(assoc nested-map :attributes (assoc (:attributes nested-map) :intelligence {:a 10 :b 20}))
(assoc nested-map :attributes (assoc (:attributes nested-map) :intelligence (assoc (:intelligence (:attributes nested-map)) :spatial 10 )))
Yay, there you go
(assoc nested-map :attributes
(assoc (:attributes nested-map) :intelligence
(assoc (:intelligence (:attributes nested-map)
:spatial 10))))
Let’s throw in one more:
(def simple
{:name "Smooches McCutes"
:intelligence 10})
(assoc simple :intelligence 10)
The compare:
(assoc simple :intelligence 10)
(assoc character :attributes
(assoc (:attributes character)
:intelligence 26))
(assoc nested-map :attributes
(assoc (:attributes nested-map)
:intelligence (assoc (:intelligence (:attributes nested-map))
:spatial 60)))
yes those are three good representative cases. There is probably something about nil and identity too
Probably, but we can ignore those for now
Now, could you break that last one down into 3 functions you could call to assoc a new value into the inner :spatial
key?
I’m thinking a function to set the :attributes
key, a function to set the :intelligence
key, and one to set the :spatial
key.
“last one” == (assoc nested-map...)
Kinda, except each function is going to need to take the map it’s working with
No global variables allowed 😉
Since I can’t do this:
(defn my-assoc-in
([m k v] (assoc m k v))
([m [k & ks] v] (println "This is the same arity... bummer.")))
(defn printks [m [k & ks] v] (println ks))
(printks {:a 1 :b 2} :happy "yes")
; --> UnsupportedOperationException nth not supported on this type: Keyword clojure.lang.RT.nthFrom (RT.java:947)
For arrays just use count
vector I mean
“For vectors, just use count
”
bleh is it friday yet?
What arguments are you calling it with?
Ok, your :happy
needs to be [:happy]
You want to match the signature for assoc-in
, which takes (assoc-in coll ks v ...)
, where ks
is a vector of keys.
ks
will always be a vector, but it might be empty
What you’ve got so far looks promising
If you’re having trouble with the “recurse” part, try writing functions assoc-attributes
, assoc-intelligence
, and assoc-spatial
, so that you can combine them to update the :spatial
value.
defn my-assoc-in
[m [k & ks] v]
(if (empty? ks)
(assoc m k v)
(assoc m k (my-assoc-in (get m k {}) ks v))))
You’d be surprised how much thinking is futzing. 😉
That’s what makes the repl so useful.
No prob
Seems like update-in
will be like
(defn my-update-in
[m [k & ks] func & args]
(if (empty? ks)
(update m k func args)
(update m k (my-update-in (get m k {}) ks #(func args ???)))))
;; ?? should be something that would get the value
mostly, except args
will be a vector because of the &
, so you’ll need an apply
to spread the args back out into a normal function call
(apply func args)
that is.
oh wait, no I’m leading you astray, my bad
You want to just pass that function thru until you get down to your base case
Right, that’s where you need it
There’s actually something interesting happening when you add & args
to a recursive function.
(defn my-update-in
[m [k & ks] func & args]
(if (empty? ks)
(update m k (apply func args (get m k)))
(update m k (my-update-in (get m k {}) ks func))))
Not quite. You need to pass a function to update
Although it would be simpler if you just assoc in your base case
(assoc m k (apply func args (get m k)))
I think that would work.
Your non-base case isn’t passing the args
thru, you’ll need that.
But then you’ll run into something really wacky: every time you recurse, the args get wrapped inside a vector, so if you call (my-update-in m [:a :b :c] f 1 2 3)
, it’s going to get to the base case as (my-update-in m [:c} [[[1 2 3]]])