LeetCode 893. Groups of Special-Equivalent Strings
Two strings are special-equivalent if you can swap characters at even indices among themselves and swap characters at odd indices among themselves, any number of times. Return the number of groups of special-equivalent strings.
Examples
Example 1:
Input: words = ["abcd","cdab","cbad","xyzz","zzxy","zzyx"]
Output: 3
Explanation: Groups are ["abcd","cdab","cbad"], ["xyzz","zzxy"], ["zzyx"].
Example 2:
Input: words = ["abc","acb","bac","bca","cab","cba"]
Output: 3
Constraints
1 <= words.length <= 10001 <= words[i].length <= 20words[i]consist of lowercase English letters- All
words[i]have the same length
Thinking Process
Swapping within even positions and within odd positions means:
- Order inside even positions doesn’t matter
- Order inside odd positions doesn’t matter
So what actually matters is:
- The multiset of even-index characters
- The multiset of odd-index characters
Two strings are equivalent if and only if:
sorted(even chars) == sorted(even chars)
AND
sorted(odd chars) == sorted(odd chars)
Canonical Form
For each word, build a signature:
- Extract even-index characters
- Extract odd-index characters
- Sort both
- Concatenate with a separator
Example: "abcd" – even: ac, odd: bd – signature: "ac|bd"
All strings with the same signature belong to one group. The answer is the number of distinct signatures.
Formal View
Each string defines a pair (E_multiset, O_multiset). Swaps only permute within E and within O. So equivalence class = identical pair of multisets. This is a classic “group by canonical representation under allowed transformations” pattern, similar to Group Anagrams (LC 49).
Approach 1: Sort-Based Signature – $O(nk \log k)$
For each word, sort even-index and odd-index characters separately, concatenate into a key, and insert into a set.
class Solution {
public:
int numSpecialEquivGroups(vector<string>& words) {
unordered_set<string> groups;
for (auto& w : words) {
string even = "", odd = "";
for (int i = 0; i < (int)w.size(); i++) {
if (i % 2 == 0) even += w[i];
else odd += w[i];
}
sort(even.begin(), even.end());
sort(odd.begin(), odd.end());
groups.insert(even + "|" + odd);
}
return groups.size();
}
};
Time: $O(nk \log k)$ – sorting twice per word Space: $O(nk)$
Approach 2: Frequency Count – $O(nk)$
Since characters are lowercase letters (only 26), we can avoid sorting by counting character frequencies instead.
class Solution {
public:
int numSpecialEquivGroups(vector<string>& words) {
unordered_set<string> groups;
for (auto& w : words) {
int even[26] = {0};
int odd[26] = {0};
for (int i = 0; i < (int)w.size(); i++) {
if (i % 2 == 0) even[w[i] - 'a']++;
else odd[w[i] - 'a']++;
}
string key = "";
for (int i = 0; i < 26; i++) key += to_string(even[i]) + "#";
for (int i = 0; i < 26; i++) key += to_string(odd[i]) + "#";
groups.insert(key);
}
return groups.size();
}
};
Time: $O(nk)$ Space: $O(nk)$
Common Mistakes
- Only sorting the whole string (ignores even/odd split)
- Forgetting the parity split entirely
- Not using a separator in the key (causes hash collisions, e.g., counts
12vs1+2) - Using
vector<int>as key without a custom hash
Key Takeaways
This problem tests:
- Canonical representation – reduce equivalence to a signature
- Parity-based partitioning – even vs odd index awareness
- Frequency counting as an alternative to sorting for small alphabets
Related Problems
- 49. Group Anagrams – group by sorted canonical form
- 205. Isomorphic Strings – transformation invariant