Skip to content

Commit

Permalink
In exercises/concept/functions-generating-functions/, added:
Browse files Browse the repository at this point in the history
   .docs/instructions.md
   .docs/introduction.md
   .meta/exemplar.clj
  • Loading branch information
JaumeAmoresDS committed Nov 4, 2023
1 parent 3fe389c commit 2ba0da3
Show file tree
Hide file tree
Showing 3 changed files with 411 additions and 0 deletions.
283 changes: 283 additions & 0 deletions exercises/concept/functions-generating-functions/.docs/instructions.md
Original file line number Diff line number Diff line change
@@ -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
Original file line number Diff line number Diff line change
@@ -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
Loading

0 comments on commit 2ba0da3

Please sign in to comment.