From 2ba0da38b322337bb8b6e233125a086536c46237 Mon Sep 17 00:00:00 2001 From: Jaume Amores Date: Sat, 4 Nov 2023 12:00:30 +0000 Subject: [PATCH] In exercises/concept/functions-generating-functions/, added: .docs/instructions.md .docs/introduction.md .meta/exemplar.clj --- .../.docs/instructions.md | 283 ++++++++++++++++++ .../.docs/introduction.md | 69 +++++ .../.meta/exemplar.clj | 59 ++++ 3 files changed, 411 insertions(+) create mode 100644 exercises/concept/functions-generating-functions/.docs/instructions.md create mode 100644 exercises/concept/functions-generating-functions/.docs/introduction.md create mode 100644 exercises/concept/functions-generating-functions/.meta/exemplar.clj diff --git a/exercises/concept/functions-generating-functions/.docs/instructions.md b/exercises/concept/functions-generating-functions/.docs/instructions.md new file mode 100644 index 000000000..c9d01ae21 --- /dev/null +++ b/exercises/concept/functions-generating-functions/.docs/instructions.md @@ -0,0 +1,283 @@ +# Instructions + +Mary and John like to do frequent trips by bike. Based on their previous trips, they have estimated that their average speed is `20 km/h`. + +```clojure +(def john-mary-average-speed 20) +``` + +Based on their average speed, they want to estimate how long it will take to perform their next planned trips. + +# 1. Estimate total duration of a given trip + +Each trip is defined as a list of maps, for example: + +```clojure +(def trip + [{:origin "Paris" + :destination "Lyon" + :distance 800} + + {:origin "Lyon" + :destination "Milan" + :distance 500} + + {:origin "Milan" + :destination "Rome" + :distance 300}]) + +``` + +Using `map`, `reduce`, and `partial`, implement a function that takes as input the average speed and a given trip, and returns the estimated time that takes to do such trip: + +```clojure +(defn estimated-trip-duration + "Returns the time it takes to do the trip + given the average-speed." + [average-speed trip] + ;(reduce + (map my-partial-fn (:distance trip)))) +``` + +in our case, if we run: + +```clojure +(estimated-trip-duration john-mary-average-speed trip) ; => 80.0 +``` + +we get 80.0 + +In mathematical notation we have: + +``` +my-full-fn (speed, distance) => time it takes to travel given distance at given speed +my-partial-fn = my-full-fn (speed=average-speed, distance) => new function where the parameter `speed` is set to value `average-speed`. The returned function accepts only one parameter: +my-partial-fn (distance) => time it takes to travel given distance if speed is fixed to be `average-speed` +``` +where `average-speed` is the value introduced as parameter to `estimated-trip-duration`. + +Note that, instead of using `partial`, we could simply use an anonymous function, i.e., `#(...)`. Using `partial` will become handy in the next task where we need to nest several such functions. This is due to the fact that clojure doesn't allow to nest multiple anonymous functions, as explained [here][anonymous-functions-official-docs]. Again, we could use `fn` instead of `partial`, since we can nest multiple functions defined with `fn`. However, `partial` offers a slightly more compact syntax for achieving the same. + +## 2. Estimate duration of multiple trips + +John and Mary are still a bit undecisive about what their next trip will be. They have several potential trips in mind, but this time they have a very limited number of remaining annual leave days. They decide to estimate the total duration per trip and select the trip that takes the least amount of time. + +Implement a function that, given a list of trips, returns a list of durations: + +```clojure +(defn time-per-trip + "Returns the time it takes to do each one + of the trips in the input vector `trips`, + given `average-speed`. The result is a vector + of floats, one per trip in `trips`." + [trips average-speed] + ; (...) + ) +``` + +For example: +```clojure +(def trip1 + [{:origin "Paris" + :destination "Lyon" + :distance 800} + + {:origin "Lyon" + :destination "Milan" + :distance 500} + + {:origin "Milan" + :destination "Rome" + :distance 300}]) + +(def trip2 + [{:origin "Madrid" + :destination "Barcelona" + :distance 700} + + {:origin "Barcelona" + :destination "San Sebastian" + :distance 500} + + {:origin "San Sebastian" + :destination "Santiago de Compostela" + :distance 450}]) + +(time-per-trip [trip1 trip2] john-mary-average-speed) ; => (80.0 82.5) +``` + +While `time-per-trip` could simply call `estimated-trip-duration`. For doing this exercise, however, we recommend to avoid using `estimated-trip-duration`, so that we can see the use of partial in a nested expression. + +Hint: we can use here an outer anonymous function with the syntax `#(reduce + ...)`` and use `partial` in the inner part of this expression. + +### 3. Update the estimated average speed + +As John and Mary get more and more practice in doing this type of trips by bike, they observe that their average speed is increasing over time. They would like to use the most up-to-date estimate of their average speed to calculate the duration of the next trips. + +Use `comp` to implement a function that, given the last trips, and the next potential trips: + +- calculates a new average-speed based on the last trips +- uses the new average-speed to calculate the time it takes to do each one of the next potential trips + +In order to implement this function: + +1. implement `calculate-speed`: + +```clojure +(defn calculate-speed + "Returns the speed with each the trip + `trip-with-duration` was done" + [trip-with-duration] + (...)) +``` + +where `trip-with-duration` is a map similar to `trip` but including duration: + +```clojure +(def trip-with-duration-1 + [{:origin "Paris" + :destination "Lyon" + :distance 800 + :duration 50.0} + + {:origin "Lyon" + :destination "Milan" + :distance 500 + :duration 35.0} + + {:origin "Milan" + :destination "Rome" + :distance 300 + :duration 20.0} + ]) +``` + +```clojure +(calculate-average-speed trip-with-duration-1) ; => 15.238095238095237 +``` + +2. Implement `calculate-average-speed` + +```clojure +(defn calculate-average-speed + "Returns the average speed given the input vector + `trips-with-duration`." + [trips-with-duration] + ;(...) +) +``` + +where `trips-with-duration` is a vector of `trip-with-duration` maps + +```clojure +(def trip-with-duration-2 + [{:origin "Madrid" + :destination "Barcelona" + :distance 700 + :duration 45.0} + + {:origin "Barcelona" + :destination "San Sebastian" + :distance 500 + :duration 35.0} + + {:origin "San Sebastian" + :destination "Santiago de Compostela" + :distance 450 + :duration 33.5}]) + +(def trips-with-duration [trip-with-duration-1 trip-with-duration-2]) +``` + +```clojure +(calculate-average-speed trips-with-duration) ; => 14.887770086007972 +``` + +3. Implement `time-per-trip-with-updated-speed` + +```clojure +(defn time-per-trip-with-updated-speed + "Given a vector `last-trips` with the time it + took to do the last trips, it calculates the average + speed with which those trips were done, and uses this + average speed to estimate the time it will + take to do each one of the trips in `next-trips`. The + result is a vector of floats, one per trip in `next-trips`" + [last-trips next-trips] + ; ... + ) +``` + +```clojure +(def next-trip-1 + [{:origin "Amsterdam" + :destination "Copenhaguen" + :distance 800} + + {:origin "Copenhaguen" + :destination "Berlin" + :distance 500} + + {:origin "Berlin" + :destination "Frankfurt" + :distance 300}]) + +(def next-trip-2 + [{:origin "Helsinki" + :destination "Saint PetersBurg" + :distance 400} + + {:origin "Saint PetersBurg" + :destination "Moscow" + :distance 900} + + {:origin "Moscow" + :destination "Minsk" + :distance 750}]) + +(def next-trips [next-trip-1 next-trip-2]) + +(time-per-trip-with-updated-speed trips-with-duration next-trips) ; => (107.47076229392701 137.69691418909397) +``` + +in order to implement `time-per-trip-with-updated-speed`, make use of `comp`, `partial`, and the previous functions: `calculate-average-speed` and `time-per-trip` + +### 4. Save time by using memoize + +The function `calculate-average-speed` takes a list of previous trips, calculates the speed per trip using `calculate-speed`, and then calculates the average. When we add a new trip to the list, the calculation of speed per trip needs to be repeated for all the previous trips that were already in the list. Since this is not really an expensive computation, let us implement a new function `slow-calculate-speed` where we add an artificial delay of 0.1 seconds, and wrap that function with `memoize`. + +```clojure +(defn slow-calculate-speed + "Calculates the speed with which `trip-with-duration` + was performed. It introduces a delay of 0.1 sec in the + calculation." + [trip-with-duration] +; ... +) + +(def memoized-calculate-speed (memoize slow-calculate-speed)) +``` + +### 5. Calculate the trip duration for multiple couples + +John and Mary have joined an organized group of bike travellers who travel in couples. Each couple has estimated their average speed. Given a new trip, they want to estimate the time it takes to do such trip for each one of the couples: + +```clojure +(defn time-per-couple + "Returns the estimated time it will take to finish + the given trip for each value in `average-speeds`. + The result is a vector of floats with the same length + as `average-speeds`." + [average-speeds trip] + ; (...) +) +``` + +```clojure +(def average-speeds [20.0 25.0 15.0 30.0]) +(time-per-couple average-speeds trip1) ; => [80.0 64.0 106.66666666666667 53.333333333333336] +``` + +In order to implement `time-per-couple` we can use `juxt`. The individual functions used by `juxt` can be obtained with `partial` applied to each individual average-speed. + + +[anonymous-functions-official-docs]: https://clojure.org/guides/learn/functions#_anonymous_function_syntax \ No newline at end of file diff --git a/exercises/concept/functions-generating-functions/.docs/introduction.md b/exercises/concept/functions-generating-functions/.docs/introduction.md new file mode 100644 index 000000000..11952d66e --- /dev/null +++ b/exercises/concept/functions-generating-functions/.docs/introduction.md @@ -0,0 +1,69 @@ +# Introduction + +**Higher-order functions** [in Clojure][clojure-higher-order-functions] are functions that either accept other functions as arguments or generate new functions as their result. We introduce here four important higher-order functions: `partial`, `comp`, `juxtp` and `memoize`. + +## partial + +[partial][clojure-api-partial] allows to fix some of the parameters of the original function by giving them specific values: + +```clojure +(def inc-by-9 (partial + 9)) +(inc-by-9 5) +; => 14 +``` + +## comp + +[comp][clojure-api-comp] can be used to create a composition of any number of functions we want to compose. Functions in the composition `(comp f1 f2 f3)` are evaluated from right to left: `f3` is evaluated on the input parameters, its output is passed as input to `f2` and its output is passed in turn to `f1`. + +```clojure +(def six-times-result-sum + (comp (partial * 6) +)) +(six-times-result-sum 3 2) +; = ((partial * 6) (+ 3 2)) +; = (* 6 (+ 3 2)) +; = 30 +``` + +## memoize + +[memoize][clojure-api-memoize] makes the function store previous results so that, given the same input, the *memoized* function returns the same result without having to recompute it again. + +```clojure +; original time-consuming function +(defn my-time-consuming-fn + "Original, time-consuming function" + [x] + (Thread/sleep 2000) + (* x 2) +) + +; memoized function +(def my-memoized-fn + (memoize my-time-consuming-fn) ) + +; The first execution computes the result and stores it. +(time (my-memoized-fn 3)) +; => "Elapsed time: 2001.364052 msecs" +; => 6 + +; Subsequent calls reuse the previous computation. +(time (my-memoized-fn 3)) +; => "Elapsed time: 0.043701 msecs" +; => 6 +``` + +## juxt + +[juxt][clojure-api-juxt] applies the functions passed to it in left to right order, and ensembles the individual results in a vector: + +```clojure +; Compute the product of x by successive factors, from 2 to 5 +((juxt (partial * 2) (partial * 3) (partial * 4) (partial * 5)) 10) ; => [20 30 40 50] +``` + +[clojure-higher-order-functions]: https://clojure.org/guides/higher_order_functions +[clojure-api-partial]: https://clojure.github.io/clojure/clojure.core-api.html#clojure.core/partial +[clojure-api-comp]: https://clojure.github.io/clojure/clojure.core-api.html#clojure.core/comp +[clojure-api-memoize]: https://clojuredocs.org/clojure.core/memoize +[clojure-api-juxt]: https://clojuredocs.org/clojure.core/juxt \ No newline at end of file diff --git a/exercises/concept/functions-generating-functions/.meta/exemplar.clj b/exercises/concept/functions-generating-functions/.meta/exemplar.clj new file mode 100644 index 000000000..36177403f --- /dev/null +++ b/exercises/concept/functions-generating-functions/.meta/exemplar.clj @@ -0,0 +1,59 @@ +(ns trips-by-bike) + +(defn estimated-trip-duration + "Returns the time it takes to do the trip + given the average-speed." + [average-speed trip ] + (reduce + (map (partial * (/ 1.0 average-speed)) (map :distance trip)))) + +(defn time-per-trip + "Returns the time it takes to do each one + of the trips in the input vector `trips`, + given `average-speed`. The result is a vector + of floats, one per trip in `trips`." + [trips average-speed] + (map #(reduce + (map (partial * (/ 1.0 average-speed)) (map :distance %))) trips) + ) + +(defn calculate-speed + "Returns the speed with each the trip + `trip-with-duration` was done" + [trip-with-duration] + (/ (reduce + (map :distance trip-with-duration)) + (reduce + (map :duration trip-with-duration)))) + +(defn calculate-average-speed + "Returns the average speed given the input vector + `trips-with-duration`." + [trips-with-duration] + (/ (reduce + (map calculate-speed trips-with-duration)) (count trips-with-duration))) + +(defn time-per-trip-with-updated-speed + "Given a vector `last-trips` with the time it + took to do the last trips, it calculates the average + speed with which those trips were done, and uses this + average speed to estimate the time it will + take to do each one of the trips in `next-trips`. The + result is a vector of floats, one per trip in `next-trips`" + [last-trips next-trips] + ((comp (partial time-per-trip next-trips) calculate-average-speed) last-trips)) + +(defn slow-calculate-speed + "Calculates the speed with which `trip-with-duration` + was performed. It introduces a delay of 0.1 sec in the + calculation." + [trip-with-duration] + (Thread/sleep 10) + (/ (reduce + (map :distance trip-with-duration)) + (reduce + (map :duration trip-with-duration)))) + +(def memoized-calculate-speed (memoize slow-calculate-speed)) + +(defn time-per-couple + "Returns the estimated time it will take to finish + the given trip for each value in `average-speeds. + The result is a vector of floats with the same length + as `average-speeds`." + [average-speeds trip] + ((apply juxt (map #(partial estimated-trip-duration %) average-speeds)) trip)) +