Clojure Concurrency and Blocking With core.async
Take a deep dive into the performance problem of concurrent apps that use core.async where blocking operations are involved. If you're a fan of Clojure, you need to read this.
Join the DZone community and get the full member experience.
Join For Freethis article is an attempt to dig into the performance problem of concurrent applications using core.async in situations where blocking operations are involved. "blocking" operations happen when the running program has to wait for something happening outside it; a canonical example is issuing an http request and waiting for the remote server to respond. such operations are also sometimes called "synchronous".
the core.async library comes with many high-level features like transducers and pipelines; in this article, i want to focus on the two fundamental mechanisms it provides for launching a new computation concurrently: threads and go-blocks.
new threads can be created with (thread ...) . this call runs the body in a new thread and (immediately) returns a channel to which the result of the body will be posted. similarly, a go-block is created with (go ...) — it also launches the computation concurrently, but instead of creating a new thread it posts the computation onto a thread pool of fixed size that the library maintains for all its go-blocks. most of the article is focusing on exploring the differences between these two methods.
the go-block thread pool
in any given executing clojure process, a single thread pool is dedicated to running all go-blocks. a quick glance at the clojure source code shows that the size of this pool is 8, meaning that 8 physical threads are launched. this number is hard-coded, though it can be modified by setting the clojure.core.async.pool-size property for the jvm running the program. so 8 is the default number of threads core.async has at its disposal to implement its ad-hoc cooperative multitasking.
let's start with a cute little experiment to determine the size of the thread pool empirically; this exercise will also shed some light on the effect of blocking calls inside go-blocks:
(defn launch-n-go-blocks
[n]
(let [c (async/chan)]
(dotimes [i n]
(async/go
(thread/sleep 10)
(async/>! c i)))
(receive-n c n)))
this function launches n go-blocks, each sleeping for 10 milliseconds and then pushing a number into a shared channel. then it waits to receive all numbers from the channel and returns; the effect is to block until all the go-blocks are done. receive-in is a simple function used throughout this article:
(defn receive-n
"receive n items from the given channel and return them as a vector."
[c n]
(loop [i 0
res []]
(if (= i n)
res
(recur (inc i) (conj res (async/<!! c))))))
now let's call launch-n-go-blocks several times, with an increasing n and observe what happens:
launching 1 -> "elapsed time: 11.403985 msecs"
launching 2 -> "elapsed time: 11.050685 msecs"
launching 3 -> "elapsed time: 10.37412 msecs"
launching 4 -> "elapsed time: 10.342037 msecs"
launching 5 -> "elapsed time: 10.359517 msecs"
launching 6 -> "elapsed time: 10.409539 msecs"
launching 7 -> "elapsed time: 10.543612 msecs"
launching 8 -> "elapsed time: 10.429726 msecs"
launching 9 -> "elapsed time: 20.480441 msecs"
launching 10 -> "elapsed time: 20.442724 msecs"
launching 11 -> "elapsed time: 21.115002 msecs"
launching 12 -> "elapsed time: 21.192993 msecs"
launching 13 -> "elapsed time: 21.113135 msecs"
launching 14 -> "elapsed time: 21.376159 msecs"
launching 15 -> "elapsed time: 20.754207 msecs"
launching 16 -> "elapsed time: 20.654873 msecs"
launching 17 -> "elapsed time: 31.084513 msecs"
launching 18 -> "elapsed time: 31.152651 msecs"
ignoring the minor fluctuations in measurements, there's a very clear pattern here; let's plot it:
the reason for this behavior is the blocking nature of thread/sleep. this function blocks the current thread for the specified duration (10 ms in our case); so the go-block executing it will block the thread it's currently running on. this thread is then effectively out of the pool until the sleep finishes. the plot immediately suggests the pool size is 8; as long as 8 or fewer go-blocks are launched, they all finish within ~10 ms because they all run concurrently. as soon as we go above 8, the runtime jumps to ~20 ms because one of the go-blocks will have to wait until there's a free thread in the pool.
let's try the same experiment using thread instead of go:
(defn launch-n-threads
[n]
(let [c (async/chan)]
(dotimes [i n]
(async/thread
(thread/sleep 10)
(async/>!! c i)))
(receive-n c n)))
here, each time through the loop a new thread is launched, regardless of the number of threads already executing. all these threads can run concurrently, so the runtime plot is:
the clojure documentation and talks/presentations by developers are careful to warn against running blocking operations in go-blocks ; it's also not hard to understand why this is so by thinking a bit about the fixed thread-pool based implementation. that said, it's still useful to actually see this in action using an easy-to-understand experiment. in the next section, we'll explore the real-life performance implications of blocking inside go-blocks.
blocking i/o
the sleeping example shown earlier is artificial, but the perils of blocking inside go-blocks are real. blocking happens quite often in realistic programs, most often in the context of i/o. i/o devices tend to be significantly slower than the cpu executing our program, especially if by "i/o device" we mean a web server located half-way across the world to which we issue an http request.
so the next example is going to be a simple concurrent http client; again, two versions are studied and compared — one with go-blocks, another with threads. for this sample, we'll be using the clj-http library, which provides a simple api to issue blocking http requests. the full code is available on github .
(def url-template "https://github.com/eliben/pycparser/pull/%d")
(defn blocking-get-page [i]
(clj-http.client/get (format url-template i)))
(defn go-blocking-generator
[c start n]
(doseq [i (range start (+ start n))]
(async/go (async/>! c (blocking-get-page i)))))
when go-blocking-generator is called, it launches n go-blocks, each requesting a different page from pycparser's pull requests on github. fetching one page takes between 760 and 990 ms on my machine, depending on the exact page. when running with n=20, this version takes about 2300 ms. now let's do the same with threads:
(defn thread-blocking-generator
[c start n]
(doseq [i (range start (+ start n))]
(async/thread (async/>!! c (blocking-get-page i)))))
with n=20, this version takes only 1000 ms. as expected, all threads manage to run at the same time, which is mostly spent waiting on the remote server. in the go-blocks version, only 8 blocks run concurrently because of the thread pool size; this example should really drive home the notion of just how bad blocking i/o in go-blocks is. most of the blocks sit there waiting for the thread pool to have a vacant spot, when all they have to do is just issue a http request and wait anyway.
parallelizing cpu-bound tasks
we've seen how go-blocks interact with blocking operations; now let's examine cpu-bound tasks, which spend their time doing actual computations on the cpu rather than waiting for i/o. in an older post, i explored the effects of using threads and processes in python to parallelize a simple numeric problem . here i'll be using a similar example: naïvely factorizing a large integer.
here's the function that factorizes a number into a vector of factors:
(defn factorize
"naive factorization function; takes an integer n and returns a vector of
factors."
[n]
(if (< n 2)
[]
(loop [factors []
n n
p 2]
(cond (= n 1) factors
(= 0 (mod n p)) (recur (conj factors p) (quot n p) p)
(>= (* p p) n) (conj factors n)
(> p 2) (recur factors n (+ p 2))
:else (recur factors n (+ p 1))))))
it takes around 2.3 ms to factorize the number 29 * 982451653; i'll refer to it as mynum from now on . let's examine a few strategies of factorizing a large set of numbers in parallel. we'll start with a simple "serial" factorizer, which should also introduce the api:
(defn serial-factorizer
"simple serial factorizer."
[nums]
(zipmap nums (map factorize nums)))
each factorizer function in this sample takes a sequence of numbers and returns a new map, which maps a number to its vector of factors. if we run serial-factorizer on a sequence of 1000 mynums, it takes ~2.3 seconds; no surprises here!
now, a parallel factorizer using go-blocks:
(defn async-go-factorizer
"parallel factorizer for nums, launching n go blocks."
[nums n]
;;; push nums into an input channel; spin up n go-blocks to read from this
;;; channel and add numbers to an output channel.
(let [in-c (async/chan)
out-c (async/chan)]
(async/onto-chan in-c nums)
(dotimes [i n]
(async/go-loop []
(when-let [nextnum (async/<! in-c)]
(async/>! out-c {nextnum (factorize nextnum)})
(recur))))
(receive-n-maps out-c (count nums))))
in a pattern that should be familiar by now, this function creates a couple of local channels and spins up a number of go-blocks to read and write from these channels; the code should be self-explanatory. receive-n-maps is similar to the receive-n function we've seen earlier in the article, just with maps instead of vectors.
knowing that my machine has 8 cpu threads (4 cores, hyper-threaded), i benchmarked async-go-factorizer with n=8, and it took around 680 ms, a 3.4x speedup over the serial version.
let's try the same with threads instead of go-blocks:
(defn async-thread-factorizer
"same as async-go-factorizer, but with thread instead of go."
[nums n]
(let [in-c (async/chan)
out-c (async/chan)]
(async/onto-chan in-c nums)
(dotimes [i n]
(async/thread
(loop []
(when-let [nextnum (async/<!! in-c)]
(async/>!! out-c {nextnum (factorize nextnum)})
(recur)))))
(receive-n-maps out-c (count nums))))
the performance is pretty much the same — 680 ms for 1000 numbers with parallelism of n=8.
this is an important point! on purely cpu-bound workloads, go-blocks are no worse than threads because all the physical cores are kept busy doing useful work at all time. there's no waiting involved, so there's no opportunity to steal an idle core for a different thread. one minor gotcha is to be wary of the go-block thread pool size; if you run your program on a dual socket machine with dozens of cores, you may want to bump that number up and use a wider parallelism setting.
for completeness (and fun!) let's try a couple more methods of parallelizing this computation. the pattern in these parallel factorizers is so common that core.async has a function for it - pipeline; here's how we use it:
(defn async-with-pipeline
"parallel factorizer using async/pipeline."
[nums n]
(let [in-c (async/chan)
out-c (async/chan)]
(async/onto-chan in-c nums)
(async/pipeline n out-c (map #(hash-map % (factorize %))) in-c)
(receive-n-maps out-c (count nums))))
async/pipeline takes an input channel, output channel and a transducer, as well as the parallelism. it takes care of spinning go-blocks and connecting all the channels properly. this takes about the same amount of time as the other versions shown earlier, which isn't surprising.
finally, let's try something slightly different and use clojure's parallel fold from the clojure.core.reducers library (both fold and transducers are described in my earlier article — check it out!)
(defn conjmap
([xs x] (conj xs x))
([] {}))
(defn rfold
"parallel factorizer using r/fold."
[nums]
(r/fold conjmap (r/map #(hash-map % (factorize %)) nums)))
here we don't have to set the parallelism; r/fold determines it on its own. this approach takes 1.15 seconds on 1000 numbers, quite a bit slower than the earlier attempts. it's entirely possible that the fork-join approach used by r/fold is less efficient than the manual chunking to different threads done by the other versions.
the conclusion from this section, however, should be that for purely cpu-bound tasks it doesn't matter much whether go-blocks or explicit threads are used — the performance should be more-or-less the same. that said, realistic programs don't often spend time purely in cpu-bound tasks; the reality is usually somewhere in between — some tasks do computations, other tasks wait on things. let's see a benchmark that combines the two.
combining blocking and cpu-bound tasks
this section shows an artificial benchmark that explores how a combination of blocking and cpu-bound tasks behaves when launched on go-blocs vs. threads. the cpu bound task will be the same factorization but this time with a larger number that was carefully tuned to take about 230 ms to factorize on my machine. the blocking "task" will be (thread/sleep 250). i deliberately choose the same duration for the two kinds of tasks here to make comparisons easier, but the principle applies more generally.
here is the go-block version of the benchmark:
(defn launch-go-blocking-and-compute
[nblock ncompute]
(let [c (async/chan)]
(dotimes [i nblock]
(async/go
(thread/sleep 250)
(async/>! c i)))
(dotimes [i ncompute]
(async/go
(async/>! c (factorize mynum))))
(receive-n c (+ nblock ncompute))))
nblock is the number of blocking tasks to launch; ncompute is the number of cpu-bound tasks to launch. the rest of the code is straightforward. you can guess what the threading version looks like by now — check out the full code sample if not.
the parameter space here is pretty large; let's try 32 blocking and 16 compute tasks in parallel:
nblock=32, ncompute=16
launch-go-blocking-and-compute: 1521 ms
launch-thread-blocking-and-compute: 530 ms
the larger we set nblock, the worse the situation becomes for the go-block version:
nblock=64, ncompute=16
launch-go-blocking-and-compute: 3200 ms
launch-thread-blocking-and-compute: 530 ms
up to some limit, the threading version is only limited by ncompute, since these actually occupy the cpu cores; all the blocking tasks are run in the background and can complete at the same time (after the initial 250 ms).
the go-block version fares much worse because the blocking tasks can occupy threads while the compute tasks just wait in a queue. depending on the exact mixture of blocking and compute-bound tasks, this can range from more-or-less the same to exteremely bad for the go-blocks version. ymmv!
managing callback-hell with go-blocks
we've seen the issues that come up when mixing blocking i/o with go-blocks. the reason for this is the cooperative concurrency approach implemented by go-blocks on top of a fixed thread pool. for cooperative concurrency to work well with i/o, the language should either make the scheduler aware of the i/o calls (to be able to switch to another context while blocking) or the i/o should be non-blocking. the former requires runtime support in the language, like go; the latter is what programming environments like python (with asyncio) and node.js (with its fully non-blocking standard library) do. the same applies to clojure, where core.async is just a library without actual runtime support.
the good news is that non-blocking i/o libraries are very popular these days, and clojure has a good number of them for all the common tasks you can think of. another good news is that core.async's channels make it very easy to deal with non-blocking i/o without sliding into callback hell .
here's a code sample that uses the asynchronous mode of clj-http to repeat the concurrent http request benchmark:
(defn go-async-generator
[c start n]
(doseq [i (range start (+ start n))]
(clj-http.client/get
(format url-template i)
{:async? true}
(fn [response]
(async/go (async/>! c response)))
;; exception callback.
(fn [exc]
(throw exc)))))
when passed the {:async? true} option, clj-http.client/get does a non-blocking request with a callback for the response (and another callback for an error). our "response callback" simply spins a go-block that places the response into a channel. now another go-block (or thread) can wait on the channel to perform the next step; compare that to cascading callbacks!
the performance is good too - when run with multiple requests in parallel, this version runs as fast as the thread-launching example from earlier in the article (the full code is here ). all the get requests are launched one after another, with no blocking. when the results arrive, go-blocks patiently "park" while sending them into a channel, but this is an explicit context-switch operation, so all of them peacefully run concurrently on the underlying thread pool.
conclusion
would launching a thread inside the callback work as well in the last example? yes, i think it would. so why use go-blocks?
the reason is scalability. launching threads is fine as long as you don't have too many, and as long as the latency of the launch is not too important. threads are os constructs and have fairly heavy resource requirements - in terms of memory consumption and context-switching time. go-blocks are extremely lightweight in comparison.
therefore, if you want to serve 1000s of connections concurrently from a single machine - go-blocks are the way to go, combined with non-blocking apis. note that go-blocks use a thread pool that can use multipe cores, so this isn't just a single-core concurrent multitasking solution (such as you may encounter in node.js or python's asyncio).
if the number of concurrent tasks is not too large or blocking i/o is involved, i'd recommend using async/thread. it avoids the pitfalls of blocking i/o, and in other cases performance is the same. core.async's wonderful tools like channels and alts!! are still available, making concurrent programming much more pleasant.
however, note that clojure is a multi-environment language, and in some environments (most notably clojurescript), threads are simply unavailable. in these cases using go-blocks is your only chance at any kind of reasonable concurrency (the alternative being callbacks).
another use case for go-blocks is to implement coroutines which can be useful in some cases - such as agents in games, as a replacement for complex state machines , etc. but here again, beware of the actual scale. if it's possible to use threads, just use threads. go-blocks are trickier to use correctly and one has to be always aware of what may block, lest performance is dramatically degraded.
if there's something i'm missing, please let me know!
Published at DZone with permission of Eli Bendersky. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments