Skip to content

Commit f4cb55c

Browse files
authored
Feat--add-yacht-exercise (#803)
1 parent 4da6f15 commit f4cb55c

File tree

12 files changed

+18411
-0
lines changed

12 files changed

+18411
-0
lines changed

config.json

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1133,6 +1133,14 @@
11331133
"prerequisites": [],
11341134
"difficulty": 7
11351135
},
1136+
{
1137+
"slug": "yacht",
1138+
"name": "Yacht",
1139+
"uuid": "3abaa99e-14e5-434e-ae50-e9d549c5ae6e",
1140+
"practices": [],
1141+
"prerequisites": [],
1142+
"difficulty": 3
1143+
},
11361144
{
11371145
"slug": "kindergarten-garden",
11381146
"name": "Kindergarten Garden",
Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# Instructions
2+
3+
The dice game [Yacht][yacht] is from the same family as Poker Dice, Generala and particularly Yahtzee, of which it is a precursor.
4+
In the game, five dice are rolled and the result can be entered in any of twelve categories.
5+
The score of a throw of the dice depends on category chosen.
6+
7+
## Scores in Yacht
8+
9+
| Category | Score | Description | Example |
10+
| --------------- | ---------------------- | ---------------------------------------- | ------------------- |
11+
| Ones | 1 × number of ones | Any combination | 1 1 1 4 5 scores 3 |
12+
| Twos | 2 × number of twos | Any combination | 2 2 3 4 5 scores 4 |
13+
| Threes | 3 × number of threes | Any combination | 3 3 3 3 3 scores 15 |
14+
| Fours | 4 × number of fours | Any combination | 1 2 3 3 5 scores 0 |
15+
| Fives | 5 × number of fives | Any combination | 5 1 5 2 5 scores 15 |
16+
| Sixes | 6 × number of sixes | Any combination | 2 3 4 5 6 scores 6 |
17+
| Full House | Total of the dice | Three of one number and two of another | 3 3 3 5 5 scores 19 |
18+
| Four of a Kind | Total of the four dice | At least four dice showing the same face | 4 4 4 4 6 scores 16 |
19+
| Little Straight | 30 points | 1-2-3-4-5 | 1 2 3 4 5 scores 30 |
20+
| Big Straight | 30 points | 2-3-4-5-6 | 2 3 4 5 6 scores 30 |
21+
| Choice | Sum of the dice | Any combination | 2 3 3 4 6 scores 18 |
22+
| Yacht | 50 points | All five dice showing the same face | 4 4 4 4 4 scores 50 |
23+
24+
If the dice do not satisfy the requirements of a category, the score is zero.
25+
If, for example, _Four Of A Kind_ is entered in the _Yacht_ category, zero points are scored.
26+
A _Yacht_ scores zero if entered in the _Full House_ category.
27+
28+
## Task
29+
30+
Given a list of values for five dice and a category, your solution should return the score of the dice for that category.
31+
If the dice do not satisfy the requirements of the category your solution should return 0.
32+
You can assume that five values will always be presented, and the value of each will be between one and six inclusively.
33+
You should not assume that the dice are ordered.
34+
35+
[yacht]: https://en.wikipedia.org/wiki/Yacht_(dice_game)
Lines changed: 21 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
{
2+
"authors": [
3+
"vaeng"
4+
],
5+
"files": {
6+
"solution": [
7+
"yacht.cpp",
8+
"yacht.h"
9+
],
10+
"test": [
11+
"yacht_test.cpp"
12+
],
13+
"example": [
14+
".meta/example.cpp",
15+
".meta/example.h"
16+
]
17+
},
18+
"blurb": "Score a single throw of dice in the game Yacht.",
19+
"source": "James Kilfiger, using wikipedia",
20+
"source_url": "https://en.wikipedia.org/wiki/Yacht_(dice_game)"
21+
}
Lines changed: 89 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
#include "yacht.h"
2+
3+
#include <algorithm>
4+
#include <numeric>
5+
#include <stdexcept>
6+
7+
namespace yacht {
8+
9+
int simple_scores(const std::array<int, 5>& dice, int target) {
10+
return std::accumulate(
11+
dice.begin(), dice.end(), 0,
12+
[target](int acc, int i) { return i == target ? acc + i : acc; });
13+
}
14+
15+
bool is_yacht(const std::array<int, 5>& dice) {
16+
return std::all_of(dice.begin(), dice.end(),
17+
[&dice](int i) { return i == dice[0]; });
18+
}
19+
20+
bool is_full_house(const std::array<int, 5>& dice) {
21+
const int min{dice.at(0)};
22+
const int max{dice.at(4)};
23+
if (min == max) return false;
24+
std::array<int, 5> full_house{min, min, max, max, max};
25+
if (dice == full_house) return true;
26+
full_house[2] = min;
27+
return dice == full_house;
28+
}
29+
30+
int four_of_a_kind_count(const std::array<int, 5>& dice) {
31+
const int min{dice.at(0)};
32+
const int max{dice.at(4)};
33+
int count_min{};
34+
int count_max{};
35+
for (auto element : dice) {
36+
if (element == min) ++count_min;
37+
if (element == max) ++count_max;
38+
}
39+
if (count_min >= 4) return min * 4;
40+
if (count_max >= 4) return max * 4;
41+
return 0;
42+
}
43+
44+
int score(std::array<int, 5> dice, const std::string& category) {
45+
if (category == "yacht") {
46+
return is_yacht(dice) ? 50 : 0;
47+
}
48+
if (category == "ones") {
49+
return simple_scores(dice, 1);
50+
}
51+
if (category == "twos") {
52+
return simple_scores(dice, 2);
53+
}
54+
if (category == "threes") {
55+
return simple_scores(dice, 3);
56+
}
57+
if (category == "fours") {
58+
return simple_scores(dice, 4);
59+
}
60+
if (category == "fives") {
61+
return simple_scores(dice, 5);
62+
}
63+
if (category == "sixes") {
64+
return simple_scores(dice, 6);
65+
}
66+
// the rest of the categories can make use of a sorted array:
67+
std::sort(dice.begin(), dice.end());
68+
if (category == "four of a kind") {
69+
return four_of_a_kind_count(dice);
70+
}
71+
if (category == "little straight") {
72+
std::array<int, 5> little_straight{1, 2, 3, 4, 5};
73+
return dice == little_straight ? 30 : 0;
74+
}
75+
if (category == "big straight") {
76+
std::array<int, 5> big_straight{2, 3, 4, 5, 6};
77+
return dice == big_straight ? 30 : 0;
78+
}
79+
// the last two categories need the sum of the dice:
80+
int total = std::reduce(dice.begin(), dice.end());
81+
if (category == "full house") {
82+
return is_full_house(dice) ? total : 0;
83+
}
84+
if (category == "choice") {
85+
return total;
86+
}
87+
throw std::invalid_argument("unknown category: " + category);
88+
}
89+
} // namespace yacht
Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#pragma once
2+
3+
#include <array>
4+
#include <string>
5+
6+
namespace yacht {
7+
int score(std::array<int, 5> dice, const std::string& category);
8+
9+
} // namespace yacht
Lines changed: 97 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
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+
[3060e4a5-4063-4deb-a380-a630b43a84b6]
13+
description = "Yacht"
14+
15+
[15026df2-f567-482f-b4d5-5297d57769d9]
16+
description = "Not Yacht"
17+
18+
[36b6af0c-ca06-4666-97de-5d31213957a4]
19+
description = "Ones"
20+
21+
[023a07c8-6c6e-44d0-bc17-efc5e1b8205a]
22+
description = "Ones, out of order"
23+
24+
[7189afac-cccd-4a74-8182-1cb1f374e496]
25+
description = "No ones"
26+
27+
[793c4292-dd14-49c4-9707-6d9c56cee725]
28+
description = "Twos"
29+
30+
[dc41bceb-d0c5-4634-a734-c01b4233a0c6]
31+
description = "Fours"
32+
33+
[f6125417-5c8a-4bca-bc5b-b4b76d0d28c8]
34+
description = "Yacht counted as threes"
35+
36+
[464fc809-96ed-46e4-acb8-d44e302e9726]
37+
description = "Yacht of 3s counted as fives"
38+
39+
[d054227f-3a71-4565-a684-5c7e621ec1e9]
40+
description = "Fives"
41+
42+
[e8a036e0-9d21-443a-8b5f-e15a9e19a761]
43+
description = "Sixes"
44+
45+
[51cb26db-6b24-49af-a9ff-12f53b252eea]
46+
description = "Full house two small, three big"
47+
48+
[1822ca9d-f235-4447-b430-2e8cfc448f0c]
49+
description = "Full house three small, two big"
50+
51+
[b208a3fc-db2e-4363-a936-9e9a71e69c07]
52+
description = "Two pair is not a full house"
53+
54+
[b90209c3-5956-445b-8a0b-0ac8b906b1c2]
55+
description = "Four of a kind is not a full house"
56+
57+
[32a3f4ee-9142-4edf-ba70-6c0f96eb4b0c]
58+
description = "Yacht is not a full house"
59+
60+
[b286084d-0568-4460-844a-ba79d71d79c6]
61+
description = "Four of a Kind"
62+
63+
[f25c0c90-5397-4732-9779-b1e9b5f612ca]
64+
description = "Yacht can be scored as Four of a Kind"
65+
66+
[9f8ef4f0-72bb-401a-a871-cbad39c9cb08]
67+
description = "Full house is not Four of a Kind"
68+
69+
[b4743c82-1eb8-4a65-98f7-33ad126905cd]
70+
description = "Little Straight"
71+
72+
[7ac08422-41bf-459c-8187-a38a12d080bc]
73+
description = "Little Straight as Big Straight"
74+
75+
[97bde8f7-9058-43ea-9de7-0bc3ed6d3002]
76+
description = "Four in order but not a little straight"
77+
78+
[cef35ff9-9c5e-4fd2-ae95-6e4af5e95a99]
79+
description = "No pairs but not a little straight"
80+
81+
[fd785ad2-c060-4e45-81c6-ea2bbb781b9d]
82+
description = "Minimum is 1, maximum is 5, but not a little straight"
83+
84+
[35bd74a6-5cf6-431a-97a3-4f713663f467]
85+
description = "Big Straight"
86+
87+
[87c67e1e-3e87-4f3a-a9b1-62927822b250]
88+
description = "Big Straight as little straight"
89+
90+
[c1fa0a3a-40ba-4153-a42d-32bc34d2521e]
91+
description = "No pairs but not a big straight"
92+
93+
[207e7300-5d10-43e5-afdd-213e3ac8827d]
94+
description = "Choice"
95+
96+
[b524c0cf-32d2-4b40-8fb3-be3500f3f135]
97+
description = "Yacht as choice"
Lines changed: 64 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
# Get the exercise name from the current directory
2+
get_filename_component(exercise ${CMAKE_CURRENT_SOURCE_DIR} NAME)
3+
4+
# Basic CMake project
5+
cmake_minimum_required(VERSION 3.5.1)
6+
7+
# Name the project after the exercise
8+
project(${exercise} CXX)
9+
10+
# Get a source filename from the exercise name by replacing -'s with _'s
11+
string(REPLACE "-" "_" file ${exercise})
12+
13+
# Implementation could be only a header
14+
if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/${file}.cpp)
15+
set(exercise_cpp ${file}.cpp)
16+
else()
17+
set(exercise_cpp "")
18+
endif()
19+
20+
# Use the common Catch library?
21+
if(EXERCISM_COMMON_CATCH)
22+
# For Exercism track development only
23+
add_executable(${exercise} ${file}_test.cpp ${exercise_cpp} ${file}.h $<TARGET_OBJECTS:catchlib>)
24+
elseif(EXERCISM_TEST_SUITE)
25+
# The Exercism test suite is being run, the Docker image already
26+
# includes a pre-built version of Catch.
27+
find_package(Catch2 REQUIRED)
28+
add_executable(${exercise} ${file}_test.cpp ${exercise_cpp} ${file}.h)
29+
target_link_libraries(${exercise} PRIVATE Catch2::Catch2WithMain)
30+
# When Catch is installed system wide we need to include a different
31+
# header, we need this define to use the correct one.
32+
target_compile_definitions(${exercise} PRIVATE EXERCISM_TEST_SUITE)
33+
else()
34+
# Build executable from sources and headers
35+
add_executable(${exercise} ${file}_test.cpp ${exercise_cpp} ${file}.h test/tests-main.cpp)
36+
endif()
37+
38+
set_target_properties(${exercise} PROPERTIES
39+
CXX_STANDARD 17
40+
CXX_STANDARD_REQUIRED OFF
41+
CXX_EXTENSIONS OFF
42+
)
43+
44+
set(CMAKE_BUILD_TYPE Debug)
45+
46+
if("${CMAKE_CXX_COMPILER_ID}" MATCHES "(GNU|Clang)")
47+
set_target_properties(${exercise} PROPERTIES
48+
COMPILE_FLAGS "-Wall -Wextra -Wpedantic -Werror"
49+
)
50+
endif()
51+
52+
# Configure to run all the tests?
53+
if(${EXERCISM_RUN_ALL_TESTS})
54+
target_compile_definitions(${exercise} PRIVATE EXERCISM_RUN_ALL_TESTS)
55+
endif()
56+
57+
# Tell MSVC not to warn us about unchecked iterators in debug builds
58+
if(${MSVC})
59+
set_target_properties(${exercise} PROPERTIES
60+
COMPILE_DEFINITIONS_DEBUG _SCL_SECURE_NO_WARNINGS)
61+
endif()
62+
63+
# Run the tests on every build
64+
add_custom_target(test_${exercise} ALL DEPENDS ${exercise} COMMAND ${exercise})

0 commit comments

Comments
 (0)