Sometimes people repeat letters to represent extra feeling. For example:
"hello" -> "heeellooo"
"hi" -> "hiiii"
In these strings like "heeellooo"
, we have groups of adjacent letters that are all the same: "h"
, "eee"
, "ll"
, "ooo"
.
You are given a string s
and an array of query strings words
. A query word is stretchy if it can be made to be equal to s
by any number of applications of the following extension operation: choose a group consisting of characters c
, and add some number of characters c
to the group so that the size of the group is three or more.
"hello"
, we could do an extension on the group "o"
to get "hellooo"
, but we cannot get "helloo"
since the group "oo"
has a size less than three. Also, we could do another extension like "ll" -> "lllll"
to get "helllllooo"
. If s = "helllllooo"
, then the query word "hello"
would be stretchy because of these two extension operations: query = "hello" -> "hellooo" -> "helllllooo" = s
.Return the number of query strings that are stretchy.
Example 1:
Input: s = "heeellooo", words = ["hello", "hi", "helo"] Output: 1 Explanation: We can extend "e" and "o" in the word "hello" to get "heeellooo". We can't extend "helo" to get "heeellooo" because the group "ll" is not size 3 or more.
Example 2:
Input: s = "zzzzzyyyyy", words = ["zzyy","zy","zyy"] Output: 3
Constraints:
1 <= s.length, words.length <= 100
1 <= words[i].length <= 100
s
and words[i]
consist of lowercase letters.This problem asks us to determine how many words from a given list (words
) can be "stretched" to match a target string (s
). Stretching involves extending groups of consecutive identical characters in a word to have a length of 3 or more.
The core idea is to compare the target string s
with each word in words
by iterating through both simultaneously. We maintain pointers for both strings and check character groups. If a character group in s
is stretchable (length >= 3) or has the same length as its counterpart in the word, we can continue the comparison. Otherwise, the word is not stretchy.
check(s, t)
function: This helper function compares the target string s
with a word t
.
t
exceeds s
, it cannot be stretched to match, so we return false
.i
and j
, to iterate over s
and t
, respectively.s[i]
and t[j]
are different, the words don't match, and we return false
.c1
and c2
) of their consecutive identical character groups in s
and t
respectively.s
is stretchable:
c1 < c2
(group in s
is shorter than in t
), it's not stretchable, so return false
.c1 < 3
and c1 != c2
(group in s
is shorter than 3 and different from t
), it's not stretchable, return false
.i
and j
to the end of the respective character groups.i == m && j == n
), the word t
is stretchy, and we return true
. Otherwise, return false
.Main Function:
t
in words
.check(s, t)
to see if it's stretchy.ans
) if check(s, t)
returns true
.ans
.s
, m is the number of words in words
, and wi is the length of each word. The dominant factor is the nested iteration in the check
function, which iterates through the characters of both strings.def expressiveWords(s, words):
def check(s, t):
m, n = len(s), len(t)
if n > m:
return False
i = j = 0
while i < m and j < n:
if s[i] != t[j]:
return False
k = i
while k < m and s[k] == s[i]:
k += 1
c1 = k - i
i = k
k = j
while k < n and t[k] == t[j]:
k += 1
c2 = k - j
j = k
if c1 < c2 or (c1 < 3 and c1 != c2):
return False
return i == m and j == n
ans = 0
for t in words:
if check(s, t):
ans += 1
return ans
The code in other languages (Java, C++, Go) would follow a very similar structure, implementing the check
function and the main loop according to the algorithm described above. The core logic and complexity remain the same across all languages.