-
Notifications
You must be signed in to change notification settings - Fork 67
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Applicative do #63
Applicative do #63
Conversation
Just fixed a couple bugs. The code is still hairy but I'm going to improve the naming of functions and refactor a bit later today. I've tested it with I think there may be some bugs still and I'd like to add more test cases, help if welcome in this regard. |
Sorry for the dumb question, but glancing at the tests, I can't discern the practical difference between |
Great question! One limitation of monadic bind is that all the steps are strictly sequential and happen one at a time. This piece of code: (bind (just 1)
(fn [a]
(bind (just 41)
(fn [b]
(return (+ a b))))))
;; => #<Just 42> In the first call to
(mlet [a (just 1)
b (just 41)]
(return (+ a b)))
;; => #<Just 42> Now let's see the equivalent (alet [a (just 1)
b (just 41)]
(+ a b))
;; => #<Just 42> Note that no (fapply (fn [a]
(fn [b]
(+ a b)))
(just 1)
(just 41))
;; => #<Just 42> Note that now This makes no difference at all for Maybe, but applicatives that have latency in their calculations (for example promises that do an async computation) get a pretty good evaluation strategy, which can minimize overall latency. This is similar to what Manifold's I recommend taking a look into Haxl library since it's what motivated the patch to GHC for adding applicative do. I'll be adding documentation for this pull request with more details and may write a blog post disecting what the |
Aha! I'm familiar with the proposal/patch and Haxl in the Haskell world, and thanks to your description (and rereading the patch description) I think I get it in the cats context now. A blog post sounds great! I've run into a (in-ns 'cats.core)
(require '[cats.monad.maybe :as maybe])
(def ten-million 10000000) (time
(mlet [x (maybe/just (nth (iterate inc 6) ten-million))
y (maybe/just (nth (iterate inc 7) ten-million))
x (maybe/just (- x ten-million))
y (maybe/just (- y ten-million))]
(return (* x y))))
;; "Elapsed time: 773.020927 msecs"
;; => #<Just 42> If I understand correctly, the same form with (time
(alet [x (maybe/just (nth (iterate inc 6) ten-million))
y (maybe/just (nth (iterate inc 7) ten-million))
x (maybe/just (- x ten-million))
y (maybe/just (- y ten-million))]
(* x y)))
;; "Elapsed time: 771.776612 msecs"
;; => #<Just 60000042> |
We still have to start the funcool blog but this is definitely worth writing about.
Yes, you're right. This is apparently a bug in the renaming of symbols in the body, nice to have an additional test case, thanks!
Take into account that The benefits of (require '[cats.core :as m])
(require '[promissum.core :as p])
(defn sleep-promise [wait]
(p/promise (fn [deliver]
(Thread/sleep wait)
(deliver wait))))
;; note: deref-ing for blocking the current thread waiting for the promise being delivered
(time
@(m/mlet [x (sleep-promise 42)
y (sleep-promise 41)]
(m/return (+ x y))))
;; "Elapsed time: 84.328182 msecs"
;; => 83
(time
@(m/alet [x (sleep-promise 42)
y (sleep-promise 41)]
(+ x y)))
;; "Elapsed time: 44.246427 msecs"
;; => 83 Another example for illustrating dependencies between batches: (time
@(m/mlet [x (sleep-promise 42)
y (sleep-promise 41)
z (sleep-promise (inc x))
a (sleep-promise (inc y))]
(m/return (+ z a))))
;; "Elapsed time: 194.253182 msecs"
;; => 85
(time
@(m/alet [x (sleep-promise 42)
y (sleep-promise 41)
z (sleep-promise (inc x))
a (sleep-promise (inc y))]
(+ z a)))
;; "Elapsed time: 86.20699 msecs"
;; => 85 |
Got it. I hadn't checked out promissum before. Looks cool. Does |
Should work with the canal library which implements both Applicative and Monad for core.async channels. It has received little time, we need to port to reader conditionals and make a release with the new name. |
Cool, thanks! I'll take a look. |
Then, using yurrriq/canal@efbb1cc and 486895b via lein-git-deps: (require '[clojure.core.async :refer [<!!]])
(defn sleep-chan [wait]
(m/fmap
(fn [n] (Thread/sleep n) n)
(m/pure channel-monad wait)))
(time
(<!! (m/mlet [x (sleep-chan 42)
y (sleep-chan 41)]
(m/return (+ x y)))))
;; "Elapsed time: 86.302538 msecs"
;; => 83
(time
(<!! (m/alet [x (sleep-chan 42)
y (sleep-chan 41)]
(+ x y))))
;; "Elapsed time: 87.723344 msecs"
;; => 83 Thoughts? |
It may be an issue with the channel's Applicative implementation, I'll take a close look after work. Thanks for reporting this and your time! |
I'm currently working on it and I have fixed it: (require '[cats.core :as m]
'[cats.context :as mc]
'[cats.monad.either :as either]
'[cats.builtin])
(require '[clojure.core.async :as a])
(require '[cats.labs.channel])
(defn async-call
[wait]
(a/go
(a/<! (a/timeout wait))
wait))
(defn check-mlet
[]
(time
(a/<!! (m/mlet [x (async-call 100)
y (async-call 100)]
(m/return (+ x y))))))
(defn check-alet
[]
(time
(a/<!! (m/alet [x (async-call 100)
y (async-call 100)]
(m/return (+ x y))))))
(check-alet)
;; "Elapsed time: 106.507818 msecs"
(check-mlet)
;; "Elapsed time: 202.745459 msecs" |
Awesome! |
Looks great! |
It might be good to say in the docs somewhere that in Emacs with clojure-mode, you can add this to your configure for better indentation: (define-clojure-indent
(alet 'defun)) The entirety of my cats-related config as of now, for reference: (define-clojure-indent
(alet 'defun)
(bind 'defun)
(branch 'defun)
(mlet 'defun)
(branch-left 'defun)) |
+1, going to add it to the docs now. |
I think this is ready for merge. |
Ok, just rename please the WIP commit, then I'll merge it :P |
I squashed the commit into the previous one, take a look @niwinz. |
This is a work in progress, opening the PR for review and design process.
https://ghc.haskell.org/trac/ghc/wiki/ApplicativeDo
Closes #46