1507. Reformat Date
Problem Description
The given problem requires us to take a string that represents a date in the format of Day Month Year
and convert it into the standard ISO format YYYY-MM-DD
. The Day
part of the date is represented by ordinal numbers (like "1st"
, "2nd"
, "3rd"
, and so on up to "31st"
). The Month
is given by its three-letter abbreviation (for example, "Jan"
, "Feb"
, "Mar"
, etc.), and the Year
is a four-digit number ranging from 1900 to 2100. The task is to reformat the date string so that the Year
is followed by the Month
and Day
, where the month and day are each displayed as two digits, with leading zeros if necessary.
Intuition
The intuition behind the solution is to break down the original date
string into its three components (Day
, Month
, and Year
), then reassemble the date in the required YYYY-MM-DD
format. To accomplish this, follow these steps:
- Split the original date string into its components by using the space character as a delimiter.
- Reverse the order of the components so that the
Year
comes first, followed by theMonth
, then theDay
. - Map the
Month
from its abbreviation to its corresponding month number. This is done by creating a string that contains the abbreviations in order and finding the index of each month in this string. Since each abbreviation is three characters long, divide the index by 3 and add 1 to get the month number. - Format the
Month
so that it is always displayed as two digits by padding with a leading zero if necessary. - Remove the ordinal suffix (e.g.,
"st"
,"nd"
,"rd"
,"th"
) from theDay
part and also ensure it is always two digits, adding a leading zero if needed. - Join the three components with hyphens to form the final standardized date string.
By following these logical steps, we convert the date from its given verbose form into a standardized format that is widely accepted and easy to understand programmatically.
Solution Approach
The solution is implemented in Python and follows a direct approach, leveraging Python's list and string manipulation capabilities. Here's a step-by-step explanation of the implementation:
-
The input date string is split into its different components (Day, Month, Year) using the
split
method, which uses whitespace as the default separator.1s = date.split()
-
Reverse the list
s
so thatYear
becomes the first element, followed byMonth
andDay
.1s.reverse()
-
Create a string
months
that contains all the month abbreviations concatenated together. This acts like a lookup table to easily map each month abbreviation to its numeric value.1months = " JanFebMarAprMayJunJulAugSepOctNovDec"
-
Calculate the numeric representation of the month. We find the index of the month in the
months
string usingindex
, divide it by 3 (since each month abbreviation consists of 3 characters), and add 1 because indexing starts at 1 in ourmonths
string.1s[1] = str(months.index(s[1]) // 3 + 1).zfill(2)
Use
zfill(2)
to make sure the result always has two digits, padding with a zero if necessary. -
The day component still contains the ordinal suffix, which we strip off by slicing the string excluding the last two characters (the ordinal part), and then use
zfill(2)
to ensure the day is also two digits.1s[2] = s[2][:-2].zfill(2)
-
Finally, we join the components of the date together with hyphens to form the required
YYYY-MM-DD
format, and return the result.1return "-".join(s)
This implementation does not use any complex algorithms or data structures; it's mainly targeted towards the utilization of basic string operations and list manipulation to format the date string correctly. The approach is simple, efficient, and does not require any additional libraries or resources beyond standard Python capabilities.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's take an example date string: "21st Jan 2023"
. Our goal is to convert this to the ISO format YYYY-MM-DD
.
Following the steps outlined in the solution approach:
-
Split the string into components:
1s = "21st Jan 2023".split() 2# s now contains ["21st", "Jan", "2023"]
-
Reverse the list so that the
Year
comes first:1s.reverse() 2# s now contains ["2023", "Jan", "21st"]
-
Create a string
months
to help us map theMonth
abbreviation to a number:1months = " JanFebMarAprMayJunJulAugSepOctNovDec" 2# This helps us find the numeric representation of "Jan"
-
Convert the
Month
from abbreviation (Jan
) to number (01
):1s[1] = str(months.index(s[1]) // 3).zfill(2) 2# index of "Jan" in the `months` string is 4, so the month number is (4 // 3) + 1 = 02
-
Remove the ordinal suffix from the
Day
and add a leading zero if necessary:1s[2] = s[2][:-2].zfill(2) 2# "21st" becomes "21", and since it's already two digits, no leading zero is added
-
Join the components with hyphens to get the final date string in ISO format:
1iso_date = "-".join(s) 2# iso_date is "2023-01-21"
The original date string of "21st Jan 2023"
has now been successfully converted to "2023-01-21"
using the described approach. This process can be applied to any date string in the given format to achieve the desired ISO standard date format.
Solution Implementation
1class Solution:
2 def reformatDate(self, date: str) -> str:
3 # Split the date string into a list (e.g., '20th Oct 2052' -> ['20th', 'Oct', '2052'])
4 date_components = date.split()
5
6 # Reverse the list to start with the year (e.g., ['2052', 'Oct', '20th'])
7 date_components.reverse()
8
9 # Define a string with all months abbreviated and prefixed with a space for indexing purposes
10 months_string = " JanFebMarAprMayJunJulAugSepOctNovDec"
11
12 # Find the index of the month in the months string and convert it to a string with leading zero if necessary
13 # Using `// 3` because each month is represented by three characters and we want to start at 1 for January
14 date_components[1] = str(months_string.index(date_components[1]) // 3).zfill(2)
15
16 # Remove the 'st', 'nd', 'rd', 'th' from the day part and add a leading zero if necessary
17 date_components[2] = date_components[2][:-2].zfill(2)
18
19 # Join the components with hyphens to form the reformatted date string (e.g., '2052-10-20')
20 return "-".join(date_components)
21
22
23# Example usage:
24# sol = Solution()
25# formatted_date = sol.reformatDate("20th Oct 2052")
26# print(formatted_date) # Output: "2052-10-20"
27
1class Solution {
2 public String reformatDate(String date) {
3 // Split the input date string into an array of strings
4 String[] parts = date.split(" ");
5
6 // String containing abbreviations of months for easy lookup
7 String months = " JanFebMarAprMayJunJulAugSepOctNovDec";
8
9 // Extract the day and remove the ordinal suffix (st, nd, rd, th)
10 int day = Integer.parseInt(parts[0].substring(0, parts[0].length() - 2));
11
12 // Calculate the month by finding the index of the month abbreviation in the months string
13 // Divide by 3 because each month abbreviation consists of three characters
14 // And add 1 because month index should start from 1 instead of 0
15 int month = months.indexOf(parts[1]) / 3;
16
17 // Reassemble the date in the format "YYYY-MM-DD"
18 // Use String.format to ensure leading zeros where necessary
19 return String.format("%s-%02d-%02d", parts[2], month, day);
20 }
21}
22
1#include <iostream>
2#include <sstream>
3#include <string>
4using namespace std;
5
6class Solution {
7public:
8 // Function to reformat a date string from "DDth Month YYYY" format to "YYYY-MM-DD" format
9 string reformatDate(string date) {
10 // A string containing abbreviations of all months in order for easy indexing
11 string monthsStr = " JanFebMarAprMayJunJulAugSepOctNovDec";
12
13 // Stringstream to parse the input date string
14 stringstream ss(date);
15
16 string year; // To hold the year as a string
17 string temp; // Temporary string to hold the "th", "nd", "st" suffixes in date
18 string month; // To hold the month as a string
19 int day; // To store the day as an integer
20
21 // Read and parse the date string
22 ss >> day >> temp >> month >> year;
23
24 // Find the starting position of the month in the monthsStr
25 // Divide the index by 3 as there are 3 characters per month and then add 1 to get the numerical month
26 month = to_string(monthsStr.find(month) / 3);
27
28 // Add leading zero if needed to the month
29 string formattedMonth = (month.size() == 1 ? "0" + month : month);
30
31 // Add leading zero to the day if it is less than 10
32 string formattedDay = (day > 9 ? "" : "0") + to_string(day);
33
34 // Return the reformatted date string in "YYYY-MM-DD" format
35 return year + "-" + formattedMonth + "-" + formattedDay;
36 }
37};
38
1function reformatDate(date: string): string {
2 // Split the input date string into an array
3 const dateParts = date.split(' ');
4
5 // Define a string of month abbreviations for index lookup
6 const monthAbbreviations = ' JanFebMarAprMayJunJulAugSepOctNovDec';
7
8 // Extract the day from the 'dateParts' array and parse it as an integer
9 // We remove the last two characters ('th', 'nd', 'st', 'rd') before parsing
10 const day = parseInt(dateParts[0].substring(0, dateParts[0].length - 2));
11
12 // Find the position of the month abbreviation in the 'monthAbbreviations' string,
13 // Divide by 3 because each abbreviation is 3 characters long, and add 1 (since index starts at " Jan")
14 const month = Math.floor(monthAbbreviations.indexOf(dateParts[1]) / 3);
15
16 // Reform the date in the YYYY-MM-DD format
17 // Use 'padStart' to ensure day and month are two digits
18 return `${dateParts[2]}-${month.toString().padStart(2, '0')}-${day.toString().padStart(2, '0')}`;
19}
20
Time and Space Complexity
Time Complexity
The time complexity of the code primarily involves splitting the input string, reversing the split parts, indexing into a string, and joining the parts back into a formatted string.
-
split()
: The split operation is performed once on the input string. Ifn
is the size (length) of the input string, split() would generally have a time complexity ofO(n)
as it goes through the entire string once to split based on the spaces. -
reverse()
: Reversing the list of split parts happens inO(k)
time, wherek
is the number of elements in the list after splitting, which is always 3 in this case given the date format, so we consider thisO(1)
. -
index()
: Indexing into a string to find the position of a substring, such as finding the month in themonths
string. In the worst case, this could beO(m)
, wherem
is the length of themonths
string, but sincem
is a constant (it's always 36 in this code), we can consider this operationO(1)
. -
zfill()
: Thezfill()
operation isO(d)
whered
is the max length of the string being filled. Here,d
is constant 2, so the complexity isO(1)
. -
join()
: The join operation complexity isO(n)
, based on the total length of all strings being joined, which is in this case the length of the output string, which we presume is also proportional ton
.
Considering these operations and knowing that some are constant time, we can approximate the overall time complexity as O(n)
as the split()
and join()
operations dominate the overall time.
Space Complexity
For space complexity, the main concern is any additional space aside from the input that we need to allocate for processing.
-
Split list
s
: This will takeO(k)
space wherek
is the number of parts after splitting, which for a date string is always 3, so this isO(1)
. -
Months string: The space used by the months string is a constant
O(1)
since it does not grow with the input. -
Temporary storage for transformation, such as when creating strings for
zfill()
and when usingjoin()
. These are proportional to the size of the output which will be a constant length string, so this is also consideredO(1)
space.
Hence, the overall additional space used by the algorithm is constant, or O(1)
.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the output of running the following function using input 56
?
1KEYBOARD = {
2 '2': 'abc',
3 '3': 'def',
4 '4': 'ghi',
5 '5': 'jkl',
6 '6': 'mno',
7 '7': 'pqrs',
8 '8': 'tuv',
9 '9': 'wxyz',
10}
11
12def letter_combinations_of_phone_number(digits):
13 def dfs(path, res):
14 if len(path) == len(digits):
15 res.append(''.join(path))
16 return
17
18 next_number = digits[len(path)]
19 for letter in KEYBOARD[next_number]:
20 path.append(letter)
21 dfs(path, res)
22 path.pop()
23
24 res = []
25 dfs([], res)
26 return res
27
1private static final Map<Character, char[]> KEYBOARD = Map.of(
2 '2', "abc".toCharArray(),
3 '3', "def".toCharArray(),
4 '4', "ghi".toCharArray(),
5 '5', "jkl".toCharArray(),
6 '6', "mno".toCharArray(),
7 '7', "pqrs".toCharArray(),
8 '8', "tuv".toCharArray(),
9 '9', "wxyz".toCharArray()
10);
11
12public static List<String> letterCombinationsOfPhoneNumber(String digits) {
13 List<String> res = new ArrayList<>();
14 dfs(new StringBuilder(), res, digits.toCharArray());
15 return res;
16}
17
18private static void dfs(StringBuilder path, List<String> res, char[] digits) {
19 if (path.length() == digits.length) {
20 res.add(path.toString());
21 return;
22 }
23 char next_digit = digits[path.length()];
24 for (char letter : KEYBOARD.get(next_digit)) {
25 path.append(letter);
26 dfs(path, res, digits);
27 path.deleteCharAt(path.length() - 1);
28 }
29}
30
1const KEYBOARD = {
2 '2': 'abc',
3 '3': 'def',
4 '4': 'ghi',
5 '5': 'jkl',
6 '6': 'mno',
7 '7': 'pqrs',
8 '8': 'tuv',
9 '9': 'wxyz',
10}
11
12function letter_combinations_of_phone_number(digits) {
13 let res = [];
14 dfs(digits, [], res);
15 return res;
16}
17
18function dfs(digits, path, res) {
19 if (path.length === digits.length) {
20 res.push(path.join(''));
21 return;
22 }
23 let next_number = digits.charAt(path.length);
24 for (let letter of KEYBOARD[next_number]) {
25 path.push(letter);
26 dfs(digits, path, res);
27 path.pop();
28 }
29}
30
Recommended Readings
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Runtime Overview When learning about algorithms and data structures you'll frequently encounter the term time complexity This concept is fundamental in computer science and offers insights into how long an algorithm takes to complete given a certain input size What is Time Complexity Time complexity represents the amount of time
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.