Skip to content

Commit d852bcd

Browse files
Add book-store exercise (exercism#106)
1 parent 91b7175 commit d852bcd

File tree

15 files changed

+775
-0
lines changed

15 files changed

+775
-0
lines changed

config.json

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -539,6 +539,14 @@
539539
"prerequisites": [],
540540
"difficulty": 6
541541
},
542+
{
543+
"slug": "book-store",
544+
"name": "Book Store",
545+
"uuid": "b98c9123-8c26-455f-9b1f-787fb61b5be0",
546+
"practices": [],
547+
"prerequisites": [],
548+
"difficulty": 7
549+
},
542550
{
543551
"slug": "change",
544552
"name": "Change",
Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
# Instructions append
2+
3+
## Subtypes
4+
5+
This exercise defines a Subtype called `Book`, for natural numbers 1 through 5.
6+
A Subtype can be thought of as a pair `⟨x, h⟩`, where x is the value and h is the proof of its validity.
7+
8+
The value inside a Subtype (x, in this case) can be accessed by using .val, for example, `x.val`.
9+
Its proof can be accessed by using .property, for example, `x.property`.
10+
Both can also be accessed by pattern matching, as usual.
Lines changed: 61 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
# Instructions
2+
3+
To try and encourage more sales of different books from a popular 5 book series, a bookshop has decided to offer discounts on multiple book purchases.
4+
5+
One copy of any of the five books costs $8.
6+
7+
If, however, you buy two different books, you get a 5% discount on those two books.
8+
9+
If you buy 3 different books, you get a 10% discount.
10+
11+
If you buy 4 different books, you get a 20% discount.
12+
13+
If you buy all 5, you get a 25% discount.
14+
15+
Note that if you buy four books, of which 3 are different titles, you get a 10% discount on the 3 that form part of a set, but the fourth book still costs $8.
16+
17+
Your mission is to write code to calculate the price of any conceivable shopping basket (containing only books of the same series), giving as big a discount as possible.
18+
19+
For example, how much does this basket of books cost?
20+
21+
- 2 copies of the first book
22+
- 2 copies of the second book
23+
- 2 copies of the third book
24+
- 1 copy of the fourth book
25+
- 1 copy of the fifth book
26+
27+
One way of grouping these 8 books is:
28+
29+
- 1 group of 5 (1st, 2nd,3rd, 4th, 5th)
30+
- 1 group of 3 (1st, 2nd, 3rd)
31+
32+
This would give a total of:
33+
34+
- 5 books at a 25% discount
35+
- 3 books at a 10% discount
36+
37+
Resulting in:
38+
39+
- 5 × (100% - 25%) × $8 = 5 × $6.00 = $30.00, plus
40+
- 3 × (100% - 10%) × $8 = 3 × $7.20 = $21.60
41+
42+
Which equals $51.60.
43+
44+
However, a different way to group these 8 books is:
45+
46+
- 1 group of 4 books (1st, 2nd, 3rd, 4th)
47+
- 1 group of 4 books (1st, 2nd, 3rd, 5th)
48+
49+
This would give a total of:
50+
51+
- 4 books at a 20% discount
52+
- 4 books at a 20% discount
53+
54+
Resulting in:
55+
56+
- 4 × (100% - 20%) × $8 = 4 × $6.40 = $25.60, plus
57+
- 4 × (100% - 20%) × $8 = 4 × $6.40 = $25.60
58+
59+
Which equals $51.20.
60+
61+
And $51.20 is the price with the biggest discount.
Lines changed: 45 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
namespace BookStore
2+
3+
def Book := { x : Nat // x ≥ 1 ∧ x ≤ 5 }
4+
5+
def tally (basket : List Book) : Array Nat := Id.run do
6+
let mut result := Array.replicate 6 (0 : Nat)
7+
for book in basket do
8+
result := result.set! book.val (result[book.val]! + 1)
9+
result
10+
11+
def sort (array : Array Nat) : Array Nat :=
12+
Array.qsort array (fun a b => a < b)
13+
14+
def difference (array : Array Nat) : Array Nat := Id.run do
15+
let mut result := Array.replicate 6 array[5]!
16+
result := result.set! 0 0
17+
for i in [1:6] do
18+
result := result.set! (6 - i) (array[i]! - array[i - 1]!)
19+
result
20+
21+
def adjust (array : Array Nat) : Array Nat := Id.run do
22+
let mut result := array.set! 0 0
23+
let adjustment := min array[3]! array[5]!
24+
result := result.set! 3 (array[3]! - adjustment)
25+
result := result.set! 4 (array[4]! + 2 * adjustment)
26+
result := result.set! 5 (array[5]! - adjustment)
27+
result
28+
29+
def price (array : Array Nat) : Nat :=
30+
let one := array[1]!
31+
let two := array[2]!
32+
let three := array[3]!
33+
let four := array[4]!
34+
let five := array[5]!
35+
800 * one + 1520 * two + 2160 * three + 2560 * four + 3000 * five
36+
37+
def total (basket : List Book) : Nat :=
38+
basket
39+
|> tally
40+
|> sort
41+
|> difference
42+
|> adjust
43+
|> price
44+
45+
end BookStore
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
{
2+
"authors": [
3+
"keiravillekode"
4+
],
5+
"files": {
6+
"solution": [
7+
"BookStore.lean"
8+
],
9+
"test": [
10+
"BookStoreTest.lean"
11+
],
12+
"example": [
13+
".meta/Example.lean"
14+
]
15+
},
16+
"blurb": "To try and encourage more sales of different books from a popular 5 book series, a bookshop has decided to offer discounts of multiple-book purchases.",
17+
"source": "Inspired by the harry potter kata from Cyber-Dojo.",
18+
"source_url": "https://cyber-dojo.org/creator/home"
19+
}
Lines changed: 64 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
# This is an auto-generated file.
2+
#
3+
# Regenerating this file via `configlet sync` will:
4+
# - Recreate every `description` key/value pair
5+
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
6+
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
7+
# - Preserve any other key/value pair
8+
#
9+
# As user-added comments (using the # character) will be removed when this file
10+
# is regenerated, comments can be added via a `comment` key.
11+
12+
[17146bd5-2e80-4557-ab4c-05632b6b0d01]
13+
description = "Only a single book"
14+
15+
[cc2de9ac-ff2a-4efd-b7c7-bfe0f43271ce]
16+
description = "Two of the same book"
17+
18+
[5a86eac0-45d2-46aa-bbf0-266b94393a1a]
19+
description = "Empty basket"
20+
21+
[158bd19a-3db4-4468-ae85-e0638a688990]
22+
description = "Two different books"
23+
24+
[f3833f6b-9332-4a1f-ad98-6c3f8e30e163]
25+
description = "Three different books"
26+
27+
[1951a1db-2fb6-4cd1-a69a-f691b6dd30a2]
28+
description = "Four different books"
29+
30+
[d70f6682-3019-4c3f-aede-83c6a8c647a3]
31+
description = "Five different books"
32+
33+
[78cacb57-911a-45f1-be52-2a5bd428c634]
34+
description = "Two groups of four is cheaper than group of five plus group of three"
35+
36+
[f808b5a4-e01f-4c0d-881f-f7b90d9739da]
37+
description = "Two groups of four is cheaper than groups of five and three"
38+
39+
[fe96401c-5268-4be2-9d9e-19b76478007c]
40+
description = "Group of four plus group of two is cheaper than two groups of three"
41+
42+
[68ea9b78-10ad-420e-a766-836a501d3633]
43+
description = "Two each of first four books and one copy each of rest"
44+
45+
[c0a779d5-a40c-47ae-9828-a340e936b866]
46+
description = "Two copies of each book"
47+
48+
[18fd86fe-08f1-4b68-969b-392b8af20513]
49+
description = "Three copies of first book and two each of remaining"
50+
51+
[0b19a24d-e4cf-4ec8-9db2-8899a41af0da]
52+
description = "Three each of first two books and two each of remaining books"
53+
54+
[bb376344-4fb2-49ab-ab85-e38d8354a58d]
55+
description = "Four groups of four are cheaper than two groups each of five and three"
56+
57+
[5260ddde-2703-4915-b45a-e54dbbac4303]
58+
description = "Check that groups of four are created properly even when there are more groups of three than groups of five"
59+
60+
[b0478278-c551-4747-b0fc-7e0be3158b1f]
61+
description = "One group of one and four is cheaper than one group of two and three"
62+
63+
[cf868453-6484-4ae1-9dfc-f8ee85bbde01]
64+
description = "One group of one and two plus three groups of four is cheaper than one group of each size"
Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
namespace BookStore
2+
3+
def Book := { x : Nat // x ≥ 1 ∧ x ≤ 5 }
4+
5+
def total (basket : List Book) : Nat :=
6+
sorry
7+
8+
end BookStore

0 commit comments

Comments
 (0)