3597. Partition String
Description
Given a string s
, partition it into unique segments according to the following procedure:
- Start building a segment beginning at index 0.
- Continue extending the current segment character by character until the current segment has not been seen before.
- Once the segment is unique, add it to your list of segments, mark it as seen, and begin a new segment from the next index.
- Repeat until you reach the end of
s
.
Return an array of strings segments
, where segments[i]
is the ith
segment created.
Example 1:
Input: s = "abbccccd"
Output: ["a","b","bc","c","cc","d"]
Explanation:
Index | Segment After Adding | Seen Segments | Current Segment Seen Before? | New Segment | Updated Seen Segments |
---|---|---|---|---|---|
0 | "a" | [] | No | "" | ["a"] |
1 | "b" | ["a"] | No | "" | ["a", "b"] |
2 | "b" | ["a", "b"] | Yes | "b" | ["a", "b"] |
3 | "bc" | ["a", "b"] | No | "" | ["a", "b", "bc"] |
4 | "c" | ["a", "b", "bc"] | No | "" | ["a", "b", "bc", "c"] |
5 | "c" | ["a", "b", "bc", "c"] | Yes | "c" | ["a", "b", "bc", "c"] |
6 | "cc" | ["a", "b", "bc", "c"] | No | "" | ["a", "b", "bc", "c", "cc"] |
7 | "d" | ["a", "b", "bc", "c", "cc"] | No | "" | ["a", "b", "bc", "c", "cc", "d"] |
Hence, the final output is ["a", "b", "bc", "c", "cc", "d"]
.
Example 2:
Input: s = "aaaa"
Output: ["a","aa"]
Explanation:
Index | Segment After Adding | Seen Segments | Current Segment Seen Before? | New Segment | Updated Seen Segments |
---|---|---|---|---|---|
0 | "a" | [] | No | "" | ["a"] |
1 | "a" | ["a"] | Yes | "a" | ["a"] |
2 | "aa" | ["a"] | No | "" | ["a", "aa"] |
3 | "a" | ["a", "aa"] | Yes | "a" | ["a", "aa"] |
Hence, the final output is ["a", "aa"]
.
Constraints:
1 <= s.length <= 105
s
contains only lowercase English letters.
Solutions
Solution 1: Hash Table + Simulation
We can use a hash table \(\textit{vis}\) to record the segments that have already appeared. Then, we traverse the string \(s\), building the current segment \(t\) character by character until this segment has not appeared before. Each time we construct a new segment, we add it to the result list and mark it as seen.
After the traversal, we simply return the result list.
The time complexity is \(O(n \times \sqrt{n})\), and the space complexity is \(O(n)\), where \(n\) is the length of the string \(s\).
1 2 3 4 5 6 7 8 9 10 11 12 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
Solution 2: String Hashing + Hash Table + Simulation
We can use string hashing to speed up the lookup of segments. Specifically, we can compute a hash value for each segment and store it in a hash table. In this way, we can determine in constant time whether a segment has already appeared.
In detail, we first create a string hashing class \(\textit{Hashing}\) based on the string \(s\), which supports computing the hash value of any substring. Then, we traverse the string \(s\) using two pointers \(l\) and \(r\) to represent the start and end positions of the current segment (indices starting from \(1\)). Each time we extend \(r\), we compute the hash value \(x\) of the current segment. If this hash value is not in the hash table, we add the segment to the result list and mark its hash value as seen. Otherwise, we continue to extend \(r\) until we find a new segment.
After the traversal, we return the result list.
The time complexity is \(O(n)\), and the space complexity is \(O(n)\), where \(n\) is the length of the string \(s\).
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
|