Skip to content

Commit 342b6a8

Browse files
EmmanuelBerkowiczkahgohjagdish-15
authored
Add split-second-stopwatch exercise #2968 (#2977)
* Add split-second-stopwatch exercise #2968 Add split-second-stopwatch exercise Implement the split-second-stopwatch practice exercise with: - State management (ready/running/stopped) - Time tracking for current lap and total elapsed time - Lap functionality with previous laps history - Start, stop, reset operations with proper state validation - Error handling for invalid state transitions - Time formatting in HH:MM:SS format - Mock time advancement for testing Includes reference solution, starter implementation, and comprehensive test suite covering all canonical data test cases. * Add split-second-stopwatch exercise #2968 git add . git commit -m "Add split-second-stopwatch exercise Implement the split-second-stopwatch practice exercise with: - State management (ready/running/stopped) - Time tracking for current lap and total elapsed time - Lap functionality with previous laps history - Start, stop, reset operations with proper state validation - Error handling for invalid state transitions - Time formatting in HH:MM:SS format - Mock time advancement for testing Includes reference solution, starter implementation, and comprehensive test suite covering all canonical data test cases." * Add split-second-stopwatch exercise #2968 #2977 Gradle files added to new practice exercise. Gradle/wrapper directories and files copied into split-second-stopwatch exercise as requested. * Add split-second-stopwatch exercise #2968 #2977 Prerequisites and difficulty updated in main config.json file. * Add split-second-stopwatch exercise #2968 #2977 config.json format updated. * Add split-second-stopwatch exercise #2968 #2977 Add remaining gradle files to split-second-stopwatch exercise. * Update exercises/practice/split-second-stopwatch/src/test/java/SplitSecondStopwatchTest.java Co-authored-by: Kah Goh <[email protected]> * Update exercises/practice/split-second-stopwatch/.meta/src/reference/java/SplitSecondStopwatch.java Co-authored-by: Kah Goh <[email protected]> * Add split-second-stopwatch exercise #2968 #2977 main config.json updated to reorder new practice exercise. --------- Co-authored-by: Kah Goh <[email protected]> Co-authored-by: Jagdish Prajapati <[email protected]>
1 parent 51f3dd7 commit 342b6a8

File tree

14 files changed

+1029
-0
lines changed

14 files changed

+1029
-0
lines changed

config.json

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -842,6 +842,17 @@
842842
],
843843
"difficulty": 4
844844
},
845+
{
846+
"slug": "split-second-stopwatch",
847+
"name": "Split-Second Stopwatch",
848+
"uuid": "9510c0ae-9977-4260-8991-0e8e849094b0",
849+
"practices": [],
850+
"prerequisites": [
851+
"exceptions",
852+
"if-else-statements"
853+
],
854+
"difficulty": 4
855+
},
845856
{
846857
"slug": "sum-of-multiples",
847858
"name": "Sum of Multiples",
Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# Instructions
2+
3+
Your task is to build a stopwatch to keep precise track of lap times.
4+
5+
The stopwatch uses four commands (start, stop, lap, and reset) to keep track of:
6+
7+
1. The current lap's tracked time
8+
2. Previously recorded lap times
9+
10+
What commands can be used depends on which state the stopwatch is in:
11+
12+
1. Ready: initial state
13+
2. Running: tracking time
14+
3. Stopped: not tracking time
15+
16+
| Command | Begin state | End state | Effect |
17+
| ------- | ----------- | --------- | -------------------------------------------------------- |
18+
| Start | Ready | Running | Start tracking time |
19+
| Start | Stopped | Running | Resume tracking time |
20+
| Stop | Running | Stopped | Stop tracking time |
21+
| Lap | Running | Running | Add current lap to previous laps, then reset current lap |
22+
| Reset | Stopped | Ready | Reset current lap and clear previous laps |
Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
# Introduction
2+
3+
You've always run for the thrill of it — no schedules, no timers, just the sound of your feet on the pavement.
4+
But now that you've joined a competitive running crew, things are getting serious.
5+
Training sessions are timed to the second, and every split second counts.
6+
To keep pace, you've picked up the _Split-Second Stopwatch_ — a sleek, high-tech gadget that's about to become your new best friend.
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
{
2+
"authors": [
3+
"EmmanuelBerkowicz"
4+
],
5+
"files": {
6+
"solution": [
7+
"src/main/java/SplitSecondStopwatch.java"
8+
],
9+
"test": [
10+
"src/test/java/SplitSecondStopwatchTest.java"
11+
],
12+
"example": [
13+
".meta/src/reference/java/SplitSecondStopwatch.java"
14+
]
15+
},
16+
"blurb": "Keep track of time through a digital stopwatch.",
17+
"source": "Erik Schierboom",
18+
"source_url": "https://github.com/exercism/problem-specifications/pull/2547"
19+
}
Lines changed: 125 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
import java.util.ArrayList;
2+
import java.util.List;
3+
4+
public class SplitSecondStopwatch {
5+
6+
/**
7+
* A split-second stopwatch that tracks elapsed time with lap functionality.
8+
* Supports start, stop, reset, and lap operations with precise time tracking.
9+
* Times are formatted in HH:MM:SS format with two-digit precision.
10+
*
11+
* @see <a href="https://github.com/exercism/problem-specifications/tree/main/exercises/split-second-stopwatch">
12+
* Problem Specifications
13+
* </a>
14+
*/
15+
16+
private enum State { READY, RUNNING, STOPPED }
17+
18+
private State state;
19+
private long totalCompletedLaps; // Total time from completed laps
20+
private long currentLapStart; // When current lap started
21+
private long accumulated; // Accumulated time for current lap when stopped
22+
private List<String> previousLaps;
23+
private long mockTime;
24+
25+
public SplitSecondStopwatch() {
26+
this.state = State.READY;
27+
this.totalCompletedLaps = 0;
28+
this.currentLapStart = 0;
29+
this.accumulated = 0;
30+
this.previousLaps = new ArrayList<>();
31+
this.mockTime = 0;
32+
}
33+
34+
public void start() {
35+
if (state == State.RUNNING) {
36+
throw new IllegalStateException("cannot start an already running stopwatch");
37+
}
38+
39+
currentLapStart = mockTime;
40+
state = State.RUNNING;
41+
}
42+
43+
public void stop() {
44+
if (state != State.RUNNING) {
45+
throw new IllegalStateException("cannot stop a stopwatch that is not running");
46+
}
47+
48+
accumulated += mockTime - currentLapStart;
49+
state = State.STOPPED;
50+
}
51+
52+
public void reset() {
53+
if (state != State.STOPPED) {
54+
throw new IllegalStateException("cannot reset a stopwatch that is not stopped");
55+
}
56+
57+
state = State.READY;
58+
totalCompletedLaps = 0;
59+
currentLapStart = 0;
60+
accumulated = 0;
61+
previousLaps.clear();
62+
}
63+
64+
public void lap() {
65+
if (state != State.RUNNING) {
66+
throw new IllegalStateException("cannot lap a stopwatch that is not running");
67+
}
68+
69+
long currentLapTime = getCurrentLapTime();
70+
totalCompletedLaps += currentLapTime;
71+
previousLaps.add(formatTime(currentLapTime));
72+
73+
// Reset current lap and restart
74+
accumulated = 0;
75+
currentLapStart = mockTime;
76+
}
77+
78+
public String state() {
79+
return state.name().toLowerCase();
80+
}
81+
82+
public String currentLap() {
83+
return formatTime(getCurrentLapTime());
84+
}
85+
86+
public String total() {
87+
return formatTime(totalCompletedLaps + getCurrentLapTime());
88+
}
89+
90+
public List<String> previousLaps() {
91+
return new ArrayList<>(previousLaps);
92+
}
93+
94+
public void advanceTime(String timeString) {
95+
String[] parts = timeString.split(":");
96+
long hours = Long.parseLong(parts[0]);
97+
long minutes = Long.parseLong(parts[1]);
98+
long seconds = Long.parseLong(parts[2]);
99+
100+
long milliseconds = (hours * 3600 + minutes * 60 + seconds) * 1000;
101+
mockTime += milliseconds;
102+
}
103+
104+
private long getCurrentLapTime() {
105+
switch (state) {
106+
case READY:
107+
return 0;
108+
case RUNNING:
109+
return accumulated + (mockTime - currentLapStart);
110+
case STOPPED:
111+
return accumulated;
112+
default:
113+
throw new IllegalStateException("Invalid state");
114+
}
115+
}
116+
117+
private String formatTime(long milliseconds) {
118+
long totalSeconds = milliseconds / 1000;
119+
long hours = totalSeconds / 3600;
120+
long minutes = (totalSeconds % 3600) / 60;
121+
long seconds = totalSeconds % 60;
122+
123+
return String.format("%02d:%02d:%02d", hours, minutes, seconds);
124+
}
125+
}
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+
[ddb238ea-99d4-4eaa-a81d-3c917a525a23]
13+
description = "new stopwatch starts in ready state"
14+
15+
[b19635d4-08ad-4ac3-b87f-aca10e844071]
16+
description = "new stopwatch's current lap has no elapsed time"
17+
18+
[492eb532-268d-43ea-8a19-2a032067d335]
19+
description = "new stopwatch's total has no elapsed time"
20+
21+
[8a892c1e-9ef7-4690-894e-e155a1fe4484]
22+
description = "new stopwatch does not have previous laps"
23+
24+
[5b2705b6-a584-4042-ba3a-4ab8d0ab0281]
25+
description = "start from ready state changes state to running"
26+
27+
[748235ce-1109-440b-9898-0a431ea179b6]
28+
description = "start does not change previous laps"
29+
30+
[491487b1-593d-423e-a075-aa78d449ff1f]
31+
description = "start initiates time tracking for current lap"
32+
33+
[a0a7ba2c-8db6-412c-b1b6-cb890e9b72ed]
34+
description = "start initiates time tracking for total"
35+
36+
[7f558a17-ef6d-4a5b-803a-f313af7c41d3]
37+
description = "start cannot be called from running state"
38+
39+
[32466eef-b2be-4d60-a927-e24fce52dab9]
40+
description = "stop from running state changes state to stopped"
41+
42+
[621eac4c-8f43-4d99-919c-4cad776d93df]
43+
description = "stop pauses time tracking for current lap"
44+
45+
[465bcc82-7643-41f2-97ff-5e817cef8db4]
46+
description = "stop pauses time tracking for total"
47+
48+
[b1ba7454-d627-41ee-a078-891b2ed266fc]
49+
description = "stop cannot be called from ready state"
50+
51+
[5c041078-0898-44dc-9d5b-8ebb5352626c]
52+
description = "stop cannot be called from stopped state"
53+
54+
[3f32171d-8fbf-46b6-bc2b-0810e1ec53b7]
55+
description = "start from stopped state changes state to running"
56+
57+
[626997cb-78d5-4fe8-b501-29fdef804799]
58+
description = "start from stopped state resumes time tracking for current lap"
59+
60+
[58487c53-ab26-471c-a171-807ef6363319]
61+
description = "start from stopped state resumes time tracking for total"
62+
63+
[091966e3-ed25-4397-908b-8bb0330118f8]
64+
description = "lap adds current lap to previous laps"
65+
66+
[1aa4c5ee-a7d5-4d59-9679-419deef3c88f]
67+
description = "lap resets current lap and resumes time tracking"
68+
69+
[4b46b92e-1b3f-46f6-97d2-0082caf56e80]
70+
description = "lap continues time tracking for total"
71+
72+
[ea75d36e-63eb-4f34-97ce-8c70e620bdba]
73+
description = "lap cannot be called from ready state"
74+
75+
[63731154-a23a-412d-a13f-c562f208eb1e]
76+
description = "lap cannot be called from stopped state"
77+
78+
[e585ee15-3b3f-4785-976b-dd96e7cc978b]
79+
description = "stop does not change previous laps"
80+
81+
[fc3645e2-86cf-4d11-97c6-489f031103f6]
82+
description = "reset from stopped state changes state to ready"
83+
84+
[20fbfbf7-68ad-4310-975a-f5f132886c4e]
85+
description = "reset resets current lap"
86+
87+
[00a8f7bb-dd5c-43e5-8705-3ef124007662]
88+
description = "reset clears previous laps"
89+
90+
[76cea936-6214-4e95-b6d1-4d4edcf90499]
91+
description = "reset cannot be called from ready state"
92+
93+
[ba4d8e69-f200-4721-b59e-90d8cf615153]
94+
description = "reset cannot be called from running state"
95+
96+
[0b01751a-cb57-493f-bb86-409de6e84306]
97+
description = "supports very long laps"
Lines changed: 25 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
plugins {
2+
id "java"
3+
}
4+
5+
repositories {
6+
mavenCentral()
7+
}
8+
9+
dependencies {
10+
testImplementation platform("org.junit:junit-bom:5.10.0")
11+
testImplementation "org.junit.jupiter:junit-jupiter"
12+
testImplementation "org.assertj:assertj-core:3.25.1"
13+
14+
testRuntimeOnly "org.junit.platform:junit-platform-launcher"
15+
}
16+
17+
test {
18+
useJUnitPlatform()
19+
20+
testLogging {
21+
exceptionFormat = "full"
22+
showStandardStreams = true
23+
events = ["passed", "failed", "skipped"]
24+
}
25+
}
Binary file not shown.
Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
distributionBase=GRADLE_USER_HOME
2+
distributionPath=wrapper/dists
3+
distributionUrl=https\://services.gradle.org/distributions/gradle-8.7-bin.zip
4+
validateDistributionUrl=true
5+
zipStoreBase=GRADLE_USER_HOME
6+
zipStorePath=wrapper/dists

0 commit comments

Comments
 (0)