Description
Given a time represented in the format "HH:MM", form the next closest time by reusing the current digits. There is no limit on how many times a digit can be reused.
You may assume the given input string is always valid. For example, "01:34", "12:09" are all valid. "1:34", "12:9" are all invalid.
Example 1:
Input: "19:34"
Output: "19:39"
Explanation: The next closest time choosing from digits 1, 9, 3, 4, is 19:39, which occurs 5 minutes later. It is not 19:33, because this occurs 23 hours and 59 minutes later.
Example 2:
Input: "23:59"
Output: "22:22"
Explanation: The next closest time choosing from digits 2, 3, 5, 9, is 22:22. It may be assumed that the returned time is next day's time since it is smaller than the input time numerically.
Solution
Simulation, O(1), S(1)
如果能想到把时间转换成秒数,这道题就好做多了。
Simulate the clock going forward by one minute. Each time it moves forward, if all the digits are allowed, then return the current time.
The natural way to represent the time is as an integer t in the range 0 <= t < 24 * 60. Then the hours are t / 60, the minutes are t % 60, and each digit of the hours and minutes can be found by hours / 10, hours % 10 etc.
class Solution {
public String nextClosestTime(String time) {
Set allowed = new HashSet<>();
for (int i = 0; i < time.length(); ++i) {
if (time.charAt(i) != ':') { // don't forget to exclude ':'
allowed.add(time.charAt(i) - '0');
}
}
String[] arr = time.split(":");
int second = 60 * Integer.parseInt(arr[0]) + Integer.parseInt(arr[1]);
for (int i = 0; i < 24 * 60; ++i) { // iterate until met start itself (including)
second = (++second) % (24 * 60);
int[] digits = {second / 60 / 10, second / 60 % 10
, second % 60 / 10, second % 60 % 10}; // use int[] is elegant!
boolean isValid = true;
for (int d : digits) {
if (!allowed.contains(d)) {
isValid = false;
break;
}
}
if (isValid) {
return String.format("%02d:%02d", second / 60, second % 60);
}
}
return "";
}
}
Enum, O(1), S(1)
We have up to 4 different allowed digits, which naively gives us 4 * 4 * 4 * 4 possible times. For each possible time, let's check that it can be displayed on a clock: ie., hours < 24 and mins < 60. The best possible time != start is the one with the smallest cand_elapsed = (time - start) % (24 * 60), as this represents the time that has elapsed since start, and where the modulo operation is taken to be always non-negative.
For example, if we have start = 720 (ie. noon), then times like 12:05 = 725 means that (725 - 720) % (24 * 60) = 5 seconds have elapsed; while times like 00:10 = 10 means that (10 - 720) % (24 * 60) = -710 % (24 * 60) = 730 seconds have elapsed.
Also, we should make sure to handle cand_elapsed carefully. When our current candidate time cur is equal to the given starting time, then cand_elapsed will be 0 and we should handle this case appropriately.
这里比较tricky的地方是,用到了Math.floorMod方法,使得negative % positive = positive。
class Solution {
public String nextClosestTime(String time) {
Set allowed = new HashSet<>();
for (int i = 0; i < time.length(); ++i) {
if (time.charAt(i) != ':') { // don't forget to exclude ':'
allowed.add(time.charAt(i) - '0');
}
}
String[] arr = time.split(":");
int start = 60 * Integer.parseInt(arr[0]) + Integer.parseInt(arr[1]);
int minElapsed = 24 * 60;
int closest = start; // set start as default (as design)
for (int h1 : allowed) {
for (int h2 : allowed) {
if (h1 * 10 + h2 >= 24) {
continue;
}
for (int m1 : allowed) {
for (int m2 : allowed) {
if (m1 * 10 + m2 >= 60) {
continue;
}
int curr = (h1 * 10 + h2) * 60 + m1 * 10 + m2;
if (curr == start) { // skip start itself
continue;
}
int elapsed = Math.floorMod(curr - start, 24 * 60);
if (elapsed < minElapsed) {
minElapsed = elapsed;
closest = curr;
}
}
}
}
}
return String.format("%02d:%02d", closest / 60, closest % 60);
}
}